symcpu.pas 14 KB

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