symcpu.pas 13 KB

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