symcpu.pas 14 KB

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