symcpu.pas 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564
  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 is_far:boolean;
  89. end;
  90. tcpuprocvardefclass = class of tcpuprocvardef;
  91. { tcpuprocdef }
  92. tcpuprocdef = class(ti86procdef)
  93. private
  94. { returns whether the function is far by default, i.e. whether it would be
  95. far if _all_ of the following conditions are true:
  96. - we're in a far code memory model
  97. - it has no 'near' or 'far' specifiers
  98. - it is compiled in a $F- state }
  99. function default_far:boolean;
  100. public
  101. constructor create(level:byte;doregister:boolean);override;
  102. function address_type:tdef;override;
  103. function size:asizeint;override;
  104. procedure declared_far;override;
  105. procedure declared_near;override;
  106. function is_far:boolean;
  107. end;
  108. tcpuprocdefclass = class of tcpuprocdef;
  109. tcpustringdef = class(tstringdef)
  110. end;
  111. tcpustringdefclass = class of tcpustringdef;
  112. tcpuenumdef = class(tenumdef)
  113. end;
  114. tcpuenumdefclass = class of tcpuenumdef;
  115. tcpusetdef = class(tsetdef)
  116. end;
  117. tcpusetdefclass = class of tcpusetdef;
  118. { syms }
  119. tcpulabelsym = class(tlabelsym)
  120. end;
  121. tcpulabelsymclass = class of tcpulabelsym;
  122. tcpuunitsym = class(tunitsym)
  123. end;
  124. tcpuunitsymclass = class of tcpuunitsym;
  125. tcpuprogramparasym = class(tprogramparasym)
  126. end;
  127. tcpuprogramparasymclass = class(tprogramparasym);
  128. tcpunamespacesym = class(tnamespacesym)
  129. end;
  130. tcpunamespacesymclass = class of tcpunamespacesym;
  131. tcpuprocsym = class(tprocsym)
  132. end;
  133. tcpuprocsymclass = class of tcpuprocsym;
  134. tcputypesym = class(ttypesym)
  135. end;
  136. tcpuypesymclass = class of tcputypesym;
  137. tcpufieldvarsym = class(tfieldvarsym)
  138. end;
  139. tcpufieldvarsymclass = class of tcpufieldvarsym;
  140. tcpulocalvarsym = class(tlocalvarsym)
  141. end;
  142. tcpulocalvarsymclass = class of tcpulocalvarsym;
  143. tcpuparavarsym = class(tparavarsym)
  144. end;
  145. tcpuparavarsymclass = class of tcpuparavarsym;
  146. tcpustaticvarsym = class(tstaticvarsym)
  147. end;
  148. tcpustaticvarsymclass = class of tcpustaticvarsym;
  149. tcpuabsolutevarsym = class(ti86absolutevarsym)
  150. protected
  151. procedure ppuload_platform(ppufile: tcompilerppufile); override;
  152. procedure ppuwrite_platform(ppufile: tcompilerppufile); override;
  153. public
  154. addrsegment : aword;
  155. end;
  156. tcpuabsolutevarsymclass = class of tcpuabsolutevarsym;
  157. tcpupropertysym = class(tpropertysym)
  158. end;
  159. tcpupropertysymclass = class of tcpupropertysym;
  160. tcpuconstsym = class(tconstsym)
  161. end;
  162. tcpuconstsymclass = class of tcpuconstsym;
  163. tcpuenumsym = class(tenumsym)
  164. end;
  165. tcpuenumsymclass = class of tcpuenumsym;
  166. tcpusyssym = class(tsyssym)
  167. end;
  168. tcpusyssymclass = class of tcpusyssym;
  169. const
  170. pbestrealtype : ^tdef = @s80floattype;
  171. function is_proc_far(p: tabstractprocdef): boolean;
  172. {# Returns true if p is a far proc var }
  173. function is_farprocvar(p : tdef): boolean;
  174. {# Returns true if p is a far pointer def }
  175. function is_farpointer(p : tdef) : boolean;
  176. {# Returns true if p is a huge pointer def }
  177. function is_hugepointer(p : tdef) : boolean;
  178. implementation
  179. uses
  180. globals, cpuinfo, verbose, fmodule;
  181. function is_proc_far(p: tabstractprocdef): boolean;
  182. begin
  183. if p is tcpuprocdef then
  184. result:=tcpuprocdef(p).is_far
  185. else if p is tcpuprocvardef then
  186. result:=tcpuprocvardef(p).is_far
  187. else
  188. internalerror(2014041301);
  189. end;
  190. { true if p is a far proc var }
  191. function is_farprocvar(p : tdef): boolean;
  192. begin
  193. result:=(p.typ=procvardef) and tcpuprocvardef(p).is_far;
  194. end;
  195. { true if p is a far pointer def }
  196. function is_farpointer(p : tdef) : boolean;
  197. begin
  198. result:=(p.typ=pointerdef) and (tcpupointerdef(p).x86pointertyp=x86pt_far);
  199. end;
  200. { true if p is a huge pointer def }
  201. function is_hugepointer(p : tdef) : boolean;
  202. begin
  203. result:=(p.typ=pointerdef) and (tcpupointerdef(p).x86pointertyp=x86pt_huge);
  204. end;
  205. {****************************************************************************
  206. tcpuclassrefdef
  207. ****************************************************************************}
  208. function tcpuclassrefdef.alignment:shortint;
  209. begin
  210. Result:=2;
  211. end;
  212. {****************************************************************************
  213. tcpuarraydef
  214. ****************************************************************************}
  215. constructor tcpuarraydef.create_from_pointer(def: tpointerdef);
  216. begin
  217. huge:=tcpupointerdef(def).x86pointertyp=x86pt_huge;
  218. inherited create_from_pointer(def);
  219. end;
  220. function tcpuarraydef.getcopy: tstoreddef;
  221. begin
  222. result:=inherited;
  223. tcpuarraydef(result).huge:=huge;
  224. end;
  225. function tcpuarraydef.GetTypeName: string;
  226. begin
  227. Result:=inherited;
  228. if is_huge then
  229. Result:='Huge '+Result;
  230. end;
  231. procedure tcpuarraydef.ppuload_platform(ppufile: tcompilerppufile);
  232. begin
  233. inherited;
  234. huge:=(ppufile.getbyte<>0);
  235. end;
  236. procedure tcpuarraydef.ppuwrite_platform(ppufile: tcompilerppufile);
  237. begin
  238. inherited;
  239. ppufile.putbyte(byte(huge));
  240. end;
  241. {****************************************************************************
  242. tcpuprocdef
  243. ****************************************************************************}
  244. constructor tcpuprocdef.create(level: byte;doregister:boolean);
  245. begin
  246. inherited create(level,doregister);
  247. if (current_settings.x86memorymodel in x86_far_code_models) and
  248. ((cs_huge_code in current_settings.moduleswitches) or
  249. (cs_force_far_calls in current_settings.localswitches)) then
  250. procoptions:=procoptions+[po_far];
  251. end;
  252. function tcpuprocdef.address_type: tdef;
  253. begin
  254. if is_far then
  255. result:=voidfarpointertype
  256. else
  257. result:=voidnearpointertype;
  258. end;
  259. function tcpuprocdef.size: asizeint;
  260. begin
  261. result:=address_type.size;
  262. end;
  263. procedure tcpuprocdef.declared_far;
  264. begin
  265. if current_settings.x86memorymodel in x86_far_code_models then
  266. include(procoptions,po_far)
  267. else
  268. inherited declared_far;
  269. end;
  270. procedure tcpuprocdef.declared_near;
  271. begin
  272. if (current_settings.x86memorymodel in x86_far_code_models) and
  273. not (cs_huge_code in current_settings.moduleswitches) then
  274. exclude(procoptions,po_far)
  275. else
  276. inherited declared_near;
  277. end;
  278. function tcpuprocdef.default_far: boolean;
  279. begin
  280. if proctypeoption in [potype_proginit,potype_unitinit,potype_unitfinalize,
  281. potype_constructor,potype_destructor,
  282. potype_class_constructor,potype_class_destructor,
  283. potype_propgetter,potype_propsetter] then
  284. exit(true);
  285. if (procoptions*[po_classmethod,po_virtualmethod,po_abstractmethod,
  286. po_finalmethod,po_staticmethod,po_overridingmethod,
  287. po_external,po_public,po_interrupt])<>[] then
  288. exit(true);
  289. if is_methodpointer then
  290. exit(true);
  291. result:=not (visibility in [vis_private,vis_hidden]);
  292. end;
  293. function tcpuprocdef.is_far: boolean;
  294. begin
  295. result:=(po_exports in procoptions) or
  296. ((current_settings.x86memorymodel in x86_far_code_models) and ((po_far in procoptions) or default_far));
  297. end;
  298. {****************************************************************************
  299. tcpuprocvardef
  300. ****************************************************************************}
  301. constructor tcpuprocvardef.create(level: byte);
  302. begin
  303. inherited create(level);
  304. { procvars are always far in the far code memory models }
  305. if current_settings.x86memorymodel in x86_far_code_models then
  306. procoptions:=procoptions+[po_far];
  307. end;
  308. function tcpuprocvardef.is_far: boolean;
  309. begin
  310. { procvars are always far in the far code memory models }
  311. result:=current_settings.x86memorymodel in x86_far_code_models;
  312. end;
  313. {****************************************************************************
  314. tcpupointerdef
  315. ****************************************************************************}
  316. class function tcpupointerdef.default_x86_data_pointer_type: tx86pointertyp;
  317. begin
  318. if current_settings.x86memorymodel in x86_far_data_models then
  319. result:=x86pt_far
  320. else
  321. result:=inherited;
  322. end;
  323. function tcpupointerdef.alignment:shortint;
  324. begin
  325. { on i8086, we use 16-bit alignment for all pointer types, even far and
  326. huge (which are 4 bytes long) }
  327. result:=2;
  328. end;
  329. function tcpupointerdef.pointer_arithmetic_int_type:tdef;
  330. begin
  331. case x86pointertyp of
  332. x86pt_huge:
  333. result:=s32inttype;
  334. x86pt_far,
  335. x86pt_near,
  336. x86pt_near_cs,
  337. x86pt_near_ds,
  338. x86pt_near_ss,
  339. x86pt_near_es,
  340. x86pt_near_fs,
  341. x86pt_near_gs:
  342. result:=s16inttype;
  343. else
  344. internalerror(2016100403);
  345. end;
  346. end;
  347. function tcpupointerdef.pointer_arithmetic_uint_type:tdef;
  348. begin
  349. case x86pointertyp of
  350. x86pt_huge:
  351. result:=u32inttype;
  352. x86pt_far,
  353. x86pt_near,
  354. x86pt_near_cs,
  355. x86pt_near_ds,
  356. x86pt_near_ss,
  357. x86pt_near_es,
  358. x86pt_near_fs,
  359. x86pt_near_gs:
  360. result:=u16inttype;
  361. else
  362. internalerror(2016100403);
  363. end;
  364. end;
  365. function tcpupointerdef.pointer_subtraction_result_type:tdef;
  366. begin
  367. case x86pointertyp of
  368. x86pt_huge:
  369. result:=s32inttype;
  370. x86pt_far:
  371. result:=u16inttype;
  372. x86pt_near,
  373. x86pt_near_cs,
  374. x86pt_near_ds,
  375. x86pt_near_ss,
  376. x86pt_near_es,
  377. x86pt_near_fs,
  378. x86pt_near_gs:
  379. result:=s16inttype;
  380. else
  381. internalerror(2016100402);
  382. end;
  383. end;
  384. function tcpupointerdef.converted_pointer_to_array_range_type: tdef;
  385. begin
  386. case x86pointertyp of
  387. x86pt_huge:
  388. result:=s32inttype;
  389. x86pt_far,
  390. x86pt_near,
  391. x86pt_near_cs,
  392. x86pt_near_ds,
  393. x86pt_near_ss,
  394. x86pt_near_es,
  395. x86pt_near_fs,
  396. x86pt_near_gs:
  397. result:=s16inttype;
  398. else
  399. internalerror(2016100401);
  400. end;
  401. end;
  402. {****************************************************************************
  403. tcpuabsolutevarsym
  404. ****************************************************************************}
  405. procedure tcpuabsolutevarsym.ppuload_platform(ppufile: tcompilerppufile);
  406. begin
  407. inherited;
  408. if absseg then
  409. addrsegment:=ppufile.getaword;
  410. end;
  411. procedure tcpuabsolutevarsym.ppuwrite_platform(ppufile: tcompilerppufile);
  412. begin
  413. inherited;
  414. if absseg then
  415. ppufile.putaword(addrsegment);
  416. end;
  417. begin
  418. { used tdef classes }
  419. cfiledef:=tcpufiledef;
  420. cvariantdef:=tcpuvariantdef;
  421. cformaldef:=tcpuformaldef;
  422. cforwarddef:=tcpuforwarddef;
  423. cundefineddef:=tcpuundefineddef;
  424. cerrordef:=tcpuerrordef;
  425. cpointerdef:=tcpupointerdef;
  426. crecorddef:=tcpurecorddef;
  427. cimplementedinterface:=tcpuimplementedinterface;
  428. cobjectdef:=tcpuobjectdef;
  429. cclassrefdef:=tcpuclassrefdef;
  430. carraydef:=tcpuarraydef;
  431. corddef:=tcpuorddef;
  432. cfloatdef:=tcpufloatdef;
  433. cprocvardef:=tcpuprocvardef;
  434. cprocdef:=tcpuprocdef;
  435. cstringdef:=tcpustringdef;
  436. cenumdef:=tcpuenumdef;
  437. csetdef:=tcpusetdef;
  438. { used tsym classes }
  439. clabelsym:=tcpulabelsym;
  440. cunitsym:=tcpuunitsym;
  441. cprogramparasym:=tcpuprogramparasym;
  442. cnamespacesym:=tcpunamespacesym;
  443. cprocsym:=tcpuprocsym;
  444. ctypesym:=tcputypesym;
  445. cfieldvarsym:=tcpufieldvarsym;
  446. clocalvarsym:=tcpulocalvarsym;
  447. cparavarsym:=tcpuparavarsym;
  448. cstaticvarsym:=tcpustaticvarsym;
  449. cabsolutevarsym:=tcpuabsolutevarsym;
  450. cpropertysym:=tcpupropertysym;
  451. cconstsym:=tcpuconstsym;
  452. cenumsym:=tcpuenumsym;
  453. csyssym:=tcpusyssym;
  454. end.