@jhlagado/azm 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (301) hide show
  1. package/LICENSE +649 -0
  2. package/README.md +142 -0
  3. package/dist/src/analysis.d.ts +11 -0
  4. package/dist/src/analysis.js +41 -0
  5. package/dist/src/api-compile.d.ts +8 -0
  6. package/dist/src/api-compile.js +3 -0
  7. package/dist/src/api-tooling.d.ts +25 -0
  8. package/dist/src/api-tooling.js +21 -0
  9. package/dist/src/cli.d.ts +30 -0
  10. package/dist/src/cli.js +523 -0
  11. package/dist/src/compile.d.ts +10 -0
  12. package/dist/src/compile.js +175 -0
  13. package/dist/src/compileShared.d.ts +3 -0
  14. package/dist/src/compileShared.js +7 -0
  15. package/dist/src/diagnosticTypes.d.ts +77 -0
  16. package/dist/src/diagnosticTypes.js +53 -0
  17. package/dist/src/formats/index.d.ts +7 -0
  18. package/dist/src/formats/index.js +17 -0
  19. package/dist/src/formats/range.d.ts +17 -0
  20. package/dist/src/formats/range.js +45 -0
  21. package/dist/src/formats/types.d.ts +208 -0
  22. package/dist/src/formats/types.js +1 -0
  23. package/dist/src/formats/writeAsm80.d.ts +6 -0
  24. package/dist/src/formats/writeAsm80.js +86 -0
  25. package/dist/src/formats/writeBin.d.ts +7 -0
  26. package/dist/src/formats/writeBin.js +23 -0
  27. package/dist/src/formats/writeD8m.d.ts +9 -0
  28. package/dist/src/formats/writeD8m.js +239 -0
  29. package/dist/src/formats/writeHex.d.ts +9 -0
  30. package/dist/src/formats/writeHex.js +39 -0
  31. package/dist/src/formats/writeListing.d.ts +8 -0
  32. package/dist/src/formats/writeListing.js +83 -0
  33. package/dist/src/frontend/asm80/asmLine.d.ts +39 -0
  34. package/dist/src/frontend/asm80/asmLine.js +89 -0
  35. package/dist/src/frontend/asm80/parseAsmRawValues.d.ts +4 -0
  36. package/dist/src/frontend/asm80/parseAsmRawValues.js +94 -0
  37. package/dist/src/frontend/asm80/quoteScan.d.ts +10 -0
  38. package/dist/src/frontend/asm80/quoteScan.js +25 -0
  39. package/dist/src/frontend/ast.d.ts +376 -0
  40. package/dist/src/frontend/ast.js +1 -0
  41. package/dist/src/frontend/directiveAliases.d.ts +14 -0
  42. package/dist/src/frontend/directiveAliases.js +189 -0
  43. package/dist/src/frontend/grammarData.d.ts +14 -0
  44. package/dist/src/frontend/grammarData.js +65 -0
  45. package/dist/src/frontend/immExprUtils.d.ts +2 -0
  46. package/dist/src/frontend/immExprUtils.js +12 -0
  47. package/dist/src/frontend/parseAsmFlatDirectiveLine.d.ts +17 -0
  48. package/dist/src/frontend/parseAsmFlatDirectiveLine.js +187 -0
  49. package/dist/src/frontend/parseAsmInstruction.d.ts +3 -0
  50. package/dist/src/frontend/parseAsmInstruction.js +73 -0
  51. package/dist/src/frontend/parseAsmStatements.d.ts +6 -0
  52. package/dist/src/frontend/parseAsmStatements.js +16 -0
  53. package/dist/src/frontend/parseAsmStream.d.ts +10 -0
  54. package/dist/src/frontend/parseAsmStream.js +33 -0
  55. package/dist/src/frontend/parseAsmTopLevel.d.ts +18 -0
  56. package/dist/src/frontend/parseAsmTopLevel.js +34 -0
  57. package/dist/src/frontend/parseDiagnostics.d.ts +9 -0
  58. package/dist/src/frontend/parseDiagnostics.js +16 -0
  59. package/dist/src/frontend/parseEnum.d.ts +12 -0
  60. package/dist/src/frontend/parseEnum.js +70 -0
  61. package/dist/src/frontend/parseImm.d.ts +10 -0
  62. package/dist/src/frontend/parseImm.js +397 -0
  63. package/dist/src/frontend/parseLogicalLines.d.ts +11 -0
  64. package/dist/src/frontend/parseLogicalLines.js +94 -0
  65. package/dist/src/frontend/parseOp.d.ts +25 -0
  66. package/dist/src/frontend/parseOp.js +120 -0
  67. package/dist/src/frontend/parseOpHeader.d.ts +20 -0
  68. package/dist/src/frontend/parseOpHeader.js +32 -0
  69. package/dist/src/frontend/parseOperands.d.ts +4 -0
  70. package/dist/src/frontend/parseOperands.js +290 -0
  71. package/dist/src/frontend/parseParams.d.ts +6 -0
  72. package/dist/src/frontend/parseParams.js +62 -0
  73. package/dist/src/frontend/parseParserRecovery.d.ts +12 -0
  74. package/dist/src/frontend/parseParserRecovery.js +17 -0
  75. package/dist/src/frontend/parseParserShared.d.ts +2 -0
  76. package/dist/src/frontend/parseParserShared.js +8 -0
  77. package/dist/src/frontend/parseRawDataDirectiveStart.d.ts +1 -0
  78. package/dist/src/frontend/parseRawDataDirectiveStart.js +3 -0
  79. package/dist/src/frontend/parseRawDataDirectives.d.ts +7 -0
  80. package/dist/src/frontend/parseRawDataDirectives.js +142 -0
  81. package/dist/src/frontend/parseRecordFieldDecl.d.ts +34 -0
  82. package/dist/src/frontend/parseRecordFieldDecl.js +177 -0
  83. package/dist/src/frontend/parseSourceItemDispatch.d.ts +46 -0
  84. package/dist/src/frontend/parseSourceItemDispatch.js +53 -0
  85. package/dist/src/frontend/parseSourceItemTable.d.ts +16 -0
  86. package/dist/src/frontend/parseSourceItemTable.js +68 -0
  87. package/dist/src/frontend/parseTopLevelCommon.d.ts +11 -0
  88. package/dist/src/frontend/parseTopLevelCommon.js +50 -0
  89. package/dist/src/frontend/parseTypes.d.ts +24 -0
  90. package/dist/src/frontend/parseTypes.js +77 -0
  91. package/dist/src/frontend/parser.d.ts +11 -0
  92. package/dist/src/frontend/parser.js +88 -0
  93. package/dist/src/frontend/source.d.ts +28 -0
  94. package/dist/src/frontend/source.js +58 -0
  95. package/dist/src/frontend/sourceExtensions.d.ts +2 -0
  96. package/dist/src/frontend/sourceExtensions.js +6 -0
  97. package/dist/src/index.d.ts +3 -0
  98. package/dist/src/index.js +2 -0
  99. package/dist/src/lintCaseStyle.d.ts +4 -0
  100. package/dist/src/lintCaseStyle.js +129 -0
  101. package/dist/src/lowering/asmDirectiveLowering.d.ts +4 -0
  102. package/dist/src/lowering/asmDirectiveLowering.js +229 -0
  103. package/dist/src/lowering/asmDirectiveTraversal.d.ts +47 -0
  104. package/dist/src/lowering/asmDirectiveTraversal.js +52 -0
  105. package/dist/src/lowering/asmEquResolution.d.ts +8 -0
  106. package/dist/src/lowering/asmEquResolution.js +69 -0
  107. package/dist/src/lowering/asmInstructionLdHelpers.d.ts +15 -0
  108. package/dist/src/lowering/asmInstructionLdHelpers.js +102 -0
  109. package/dist/src/lowering/asmInstructionLowering.d.ts +5 -0
  110. package/dist/src/lowering/asmInstructionLowering.js +54 -0
  111. package/dist/src/lowering/asmInstructionStream.d.ts +46 -0
  112. package/dist/src/lowering/asmInstructionStream.js +51 -0
  113. package/dist/src/lowering/asmLoweringBranchCall.d.ts +43 -0
  114. package/dist/src/lowering/asmLoweringBranchCall.js +254 -0
  115. package/dist/src/lowering/asmLoweringHost.d.ts +23 -0
  116. package/dist/src/lowering/asmLoweringHost.js +1 -0
  117. package/dist/src/lowering/asmLoweringLd.d.ts +28 -0
  118. package/dist/src/lowering/asmLoweringLd.js +144 -0
  119. package/dist/src/lowering/asmRangeLowering.d.ts +17 -0
  120. package/dist/src/lowering/asmRangeLowering.js +39 -0
  121. package/dist/src/lowering/asmRawDataLowering.d.ts +16 -0
  122. package/dist/src/lowering/asmRawDataLowering.js +209 -0
  123. package/dist/src/lowering/asmSourceEmitter.d.ts +4 -0
  124. package/dist/src/lowering/asmSourceEmitter.js +9 -0
  125. package/dist/src/lowering/asmSourceInstructionLowering.d.ts +4 -0
  126. package/dist/src/lowering/asmSourceInstructionLowering.js +14 -0
  127. package/dist/src/lowering/asmUtils.d.ts +13 -0
  128. package/dist/src/lowering/asmUtils.js +105 -0
  129. package/dist/src/lowering/assemblerFlowSetup.d.ts +54 -0
  130. package/dist/src/lowering/assemblerFlowSetup.js +128 -0
  131. package/dist/src/lowering/assemblerLoweringContext.d.ts +151 -0
  132. package/dist/src/lowering/assemblerLoweringContext.js +16 -0
  133. package/dist/src/lowering/assemblerLoweringContextSplit.d.ts +7 -0
  134. package/dist/src/lowering/assemblerLoweringContextSplit.js +75 -0
  135. package/dist/src/lowering/assemblerLoweringPhases.d.ts +66 -0
  136. package/dist/src/lowering/assemblerLoweringPhases.js +166 -0
  137. package/dist/src/lowering/bytePlacement.d.ts +7 -0
  138. package/dist/src/lowering/bytePlacement.js +37 -0
  139. package/dist/src/lowering/capabilities.d.ts +67 -0
  140. package/dist/src/lowering/capabilities.js +1 -0
  141. package/dist/src/lowering/eaResolution.d.ts +58 -0
  142. package/dist/src/lowering/eaResolution.js +159 -0
  143. package/dist/src/lowering/emissionCore.d.ts +17 -0
  144. package/dist/src/lowering/emissionCore.js +21 -0
  145. package/dist/src/lowering/emit.d.ts +17 -0
  146. package/dist/src/lowering/emit.js +46 -0
  147. package/dist/src/lowering/emitContextBuilder.d.ts +63 -0
  148. package/dist/src/lowering/emitContextBuilder.js +41 -0
  149. package/dist/src/lowering/emitFinalization.d.ts +61 -0
  150. package/dist/src/lowering/emitFinalization.js +66 -0
  151. package/dist/src/lowering/emitFinalizationSetup.d.ts +19 -0
  152. package/dist/src/lowering/emitFinalizationSetup.js +26 -0
  153. package/dist/src/lowering/emitPhase1BuildProgramLoweringContext.d.ts +7 -0
  154. package/dist/src/lowering/emitPhase1BuildProgramLoweringContext.js +119 -0
  155. package/dist/src/lowering/emitPhase1Helpers.d.ts +4 -0
  156. package/dist/src/lowering/emitPhase1Helpers.js +12 -0
  157. package/dist/src/lowering/emitPhase1Types.d.ts +21 -0
  158. package/dist/src/lowering/emitPhase1Types.js +1 -0
  159. package/dist/src/lowering/emitPhase1WirePipeline.d.ts +70 -0
  160. package/dist/src/lowering/emitPhase1WirePipeline.js +203 -0
  161. package/dist/src/lowering/emitPhase1Workspace.d.ts +82 -0
  162. package/dist/src/lowering/emitPhase1Workspace.js +55 -0
  163. package/dist/src/lowering/emitPipeline.d.ts +121 -0
  164. package/dist/src/lowering/emitPipeline.js +57 -0
  165. package/dist/src/lowering/emitProgramContext.d.ts +39 -0
  166. package/dist/src/lowering/emitProgramContext.js +29 -0
  167. package/dist/src/lowering/emitState.d.ts +90 -0
  168. package/dist/src/lowering/emitState.js +124 -0
  169. package/dist/src/lowering/fixupBaseResolution.d.ts +7 -0
  170. package/dist/src/lowering/fixupBaseResolution.js +23 -0
  171. package/dist/src/lowering/fixupEmission.d.ts +64 -0
  172. package/dist/src/lowering/fixupEmission.js +199 -0
  173. package/dist/src/lowering/immMath.d.ts +2 -0
  174. package/dist/src/lowering/immMath.js +34 -0
  175. package/dist/src/lowering/inputAssets.d.ts +7 -0
  176. package/dist/src/lowering/inputAssets.js +106 -0
  177. package/dist/src/lowering/ldEncoding.d.ts +15 -0
  178. package/dist/src/lowering/ldEncoding.js +12 -0
  179. package/dist/src/lowering/ldEncodingRegMemHelpers.d.ts +5 -0
  180. package/dist/src/lowering/ldEncodingRegMemHelpers.js +124 -0
  181. package/dist/src/lowering/ldFormSelection.d.ts +26 -0
  182. package/dist/src/lowering/ldFormSelection.js +92 -0
  183. package/dist/src/lowering/ldLowering.d.ts +7 -0
  184. package/dist/src/lowering/ldLowering.js +13 -0
  185. package/dist/src/lowering/loweredAsmByteEmission.d.ts +23 -0
  186. package/dist/src/lowering/loweredAsmByteEmission.js +185 -0
  187. package/dist/src/lowering/loweredAsmPlacement.d.ts +13 -0
  188. package/dist/src/lowering/loweredAsmPlacement.js +86 -0
  189. package/dist/src/lowering/loweredAsmStreamRecording.d.ts +27 -0
  190. package/dist/src/lowering/loweredAsmStreamRecording.js +215 -0
  191. package/dist/src/lowering/loweredAsmTypes.d.ts +202 -0
  192. package/dist/src/lowering/loweredAsmTypes.js +1 -0
  193. package/dist/src/lowering/loweredFormat.d.ts +3 -0
  194. package/dist/src/lowering/loweredFormat.js +26 -0
  195. package/dist/src/lowering/loweredItemSize.d.ts +4 -0
  196. package/dist/src/lowering/loweredItemSize.js +17 -0
  197. package/dist/src/lowering/loweringDiagnostics.d.ts +9 -0
  198. package/dist/src/lowering/loweringDiagnostics.js +55 -0
  199. package/dist/src/lowering/loweringTypes.d.ts +27 -0
  200. package/dist/src/lowering/loweringTypes.js +1 -0
  201. package/dist/src/lowering/opCandidateRegistry.d.ts +9 -0
  202. package/dist/src/lowering/opCandidateRegistry.js +9 -0
  203. package/dist/src/lowering/opExpansionExecution.d.ts +15 -0
  204. package/dist/src/lowering/opExpansionExecution.js +45 -0
  205. package/dist/src/lowering/opExpansionOrchestration.d.ts +15 -0
  206. package/dist/src/lowering/opExpansionOrchestration.js +88 -0
  207. package/dist/src/lowering/opExpansionStream.d.ts +12 -0
  208. package/dist/src/lowering/opExpansionStream.js +176 -0
  209. package/dist/src/lowering/opMatching.d.ts +52 -0
  210. package/dist/src/lowering/opMatching.js +355 -0
  211. package/dist/src/lowering/opSubstitution.d.ts +13 -0
  212. package/dist/src/lowering/opSubstitution.js +175 -0
  213. package/dist/src/lowering/prescanTypes.d.ts +7 -0
  214. package/dist/src/lowering/prescanTypes.js +1 -0
  215. package/dist/src/lowering/programLowering.d.ts +144 -0
  216. package/dist/src/lowering/programLowering.js +2 -0
  217. package/dist/src/lowering/programLoweringDeclarations.d.ts +5 -0
  218. package/dist/src/lowering/programLoweringDeclarations.js +5 -0
  219. package/dist/src/lowering/programLoweringFinalize.d.ts +18 -0
  220. package/dist/src/lowering/programLoweringFinalize.js +115 -0
  221. package/dist/src/lowering/programLoweringTraversal.d.ts +2 -0
  222. package/dist/src/lowering/programLoweringTraversal.js +93 -0
  223. package/dist/src/lowering/programPrescan.d.ts +3 -0
  224. package/dist/src/lowering/programPrescan.js +37 -0
  225. package/dist/src/lowering/traceFormat.d.ts +13 -0
  226. package/dist/src/lowering/traceFormat.js +211 -0
  227. package/dist/src/pathCompare.d.ts +3 -0
  228. package/dist/src/pathCompare.js +26 -0
  229. package/dist/src/pipeline.d.ts +78 -0
  230. package/dist/src/pipeline.js +1 -0
  231. package/dist/src/registerCare/analyze.d.ts +24 -0
  232. package/dist/src/registerCare/analyze.js +327 -0
  233. package/dist/src/registerCare/annotate.d.ts +11 -0
  234. package/dist/src/registerCare/annotate.js +76 -0
  235. package/dist/src/registerCare/boundaryHints.d.ts +2 -0
  236. package/dist/src/registerCare/boundaryHints.js +10 -0
  237. package/dist/src/registerCare/carriers.d.ts +4 -0
  238. package/dist/src/registerCare/carriers.js +78 -0
  239. package/dist/src/registerCare/controlFlow.d.ts +5 -0
  240. package/dist/src/registerCare/controlFlow.js +35 -0
  241. package/dist/src/registerCare/fix.d.ts +11 -0
  242. package/dist/src/registerCare/fix.js +119 -0
  243. package/dist/src/registerCare/liveness.d.ts +7 -0
  244. package/dist/src/registerCare/liveness.js +227 -0
  245. package/dist/src/registerCare/profiles.d.ts +11 -0
  246. package/dist/src/registerCare/profiles.js +45 -0
  247. package/dist/src/registerCare/programModel.d.ts +3 -0
  248. package/dist/src/registerCare/programModel.js +181 -0
  249. package/dist/src/registerCare/report.d.ts +5 -0
  250. package/dist/src/registerCare/report.js +139 -0
  251. package/dist/src/registerCare/smartComments.d.ts +5 -0
  252. package/dist/src/registerCare/smartComments.js +247 -0
  253. package/dist/src/registerCare/sourceText.d.ts +8 -0
  254. package/dist/src/registerCare/sourceText.js +15 -0
  255. package/dist/src/registerCare/summary.d.ts +3 -0
  256. package/dist/src/registerCare/summary.js +492 -0
  257. package/dist/src/registerCare/tooling.d.ts +42 -0
  258. package/dist/src/registerCare/tooling.js +50 -0
  259. package/dist/src/registerCare/types.d.ts +154 -0
  260. package/dist/src/registerCare/types.js +1 -0
  261. package/dist/src/semantics/declVisitor.d.ts +5 -0
  262. package/dist/src/semantics/declVisitor.js +11 -0
  263. package/dist/src/semantics/env.d.ts +28 -0
  264. package/dist/src/semantics/env.js +432 -0
  265. package/dist/src/semantics/layout.d.ts +21 -0
  266. package/dist/src/semantics/layout.js +226 -0
  267. package/dist/src/semantics/layoutCastFold.d.ts +22 -0
  268. package/dist/src/semantics/layoutCastFold.js +118 -0
  269. package/dist/src/semantics/semanticsDiagnostics.d.ts +2 -0
  270. package/dist/src/semantics/semanticsDiagnostics.js +4 -0
  271. package/dist/src/semantics/typeQueries.d.ts +31 -0
  272. package/dist/src/semantics/typeQueries.js +124 -0
  273. package/dist/src/sourceIncludeExpansion.d.ts +17 -0
  274. package/dist/src/sourceIncludeExpansion.js +124 -0
  275. package/dist/src/sourceIncludePaths.d.ts +1 -0
  276. package/dist/src/sourceIncludePaths.js +12 -0
  277. package/dist/src/sourceLoader.d.ts +15 -0
  278. package/dist/src/sourceLoader.js +118 -0
  279. package/dist/src/z80/effects.d.ts +3 -0
  280. package/dist/src/z80/effects.js +516 -0
  281. package/dist/src/z80/encode.d.ts +10 -0
  282. package/dist/src/z80/encode.js +412 -0
  283. package/dist/src/z80/encodeAlu.d.ts +7 -0
  284. package/dist/src/z80/encodeAlu.js +219 -0
  285. package/dist/src/z80/encodeBitOps.d.ts +7 -0
  286. package/dist/src/z80/encodeBitOps.js +123 -0
  287. package/dist/src/z80/encodeContext.d.ts +29 -0
  288. package/dist/src/z80/encodeContext.js +1 -0
  289. package/dist/src/z80/encodeControl.d.ts +26 -0
  290. package/dist/src/z80/encodeControl.js +180 -0
  291. package/dist/src/z80/encodeCoreOps.d.ts +7 -0
  292. package/dist/src/z80/encodeCoreOps.js +131 -0
  293. package/dist/src/z80/encodeIo.d.ts +9 -0
  294. package/dist/src/z80/encodeIo.js +128 -0
  295. package/dist/src/z80/encodeLd.d.ts +13 -0
  296. package/dist/src/z80/encodeLd.js +273 -0
  297. package/dist/src/z80/encoderRegistry.d.ts +13 -0
  298. package/dist/src/z80/encoderRegistry.js +169 -0
  299. package/docs/reference/cli.md +134 -0
  300. package/docs/reference/tooling-api.md +248 -0
  301. package/package.json +98 -0
@@ -0,0 +1,185 @@
1
+ import { resolveAsmEquSymbol } from './asmEquResolution.js';
2
+ import { evalBinaryImmOp, evalUnaryImmOp } from './immMath.js';
3
+ const toByte = (value) => value & 0xff;
4
+ const toWord = (value) => value & 0xffff;
5
+ function evalLoweredImmExpr(expr, env) {
6
+ switch (expr.kind) {
7
+ case 'literal':
8
+ return expr.value;
9
+ case 'symbol': {
10
+ const direct = env.equates.get(expr.name) ?? env.enums.get(expr.name);
11
+ if (direct !== undefined)
12
+ return direct + expr.addend;
13
+ const lower = expr.name.toLowerCase();
14
+ const alt = env.equates.get(lower) ?? env.enums.get(lower);
15
+ if (alt !== undefined)
16
+ return alt + expr.addend;
17
+ const asmAlias = resolveAsmEquSymbol(expr.name, { env });
18
+ if (asmAlias !== undefined)
19
+ return asmAlias + expr.addend;
20
+ return undefined;
21
+ }
22
+ case 'unary': {
23
+ const value = evalLoweredImmExpr(expr.expr, env);
24
+ if (value === undefined)
25
+ return undefined;
26
+ return evalUnaryImmOp(expr.op, value);
27
+ }
28
+ case 'binary': {
29
+ const left = evalLoweredImmExpr(expr.left, env);
30
+ const right = evalLoweredImmExpr(expr.right, env);
31
+ if (left === undefined || right === undefined)
32
+ return undefined;
33
+ return evalBinaryImmOp(expr.op, left, right);
34
+ }
35
+ case 'opaque':
36
+ return undefined;
37
+ }
38
+ }
39
+ function blockPlacementKey(placement) {
40
+ return `base:${placement}`;
41
+ }
42
+ /** Byte length of a lowered item as emitted into a placement (matches {@link emitLoweredAsmItemBytes}). */
43
+ function loweredAsmItemEmittedSize(item, env) {
44
+ switch (item.kind) {
45
+ case 'label':
46
+ case 'const':
47
+ case 'comment':
48
+ return 0;
49
+ case 'db':
50
+ return item.values.length;
51
+ case 'dw':
52
+ return item.values.length * 2;
53
+ case 'ds': {
54
+ const size = evalLoweredImmExpr(item.size, env);
55
+ if (size === undefined || size < 0)
56
+ return 0;
57
+ return size;
58
+ }
59
+ case 'instr':
60
+ return item.bytes?.length ?? 0;
61
+ }
62
+ }
63
+ /**
64
+ * After link-time fixups are applied to `finalBytes`, patch lowered `instr` byte arrays
65
+ * (including `@raw` placeholders from abs16/rel8 fixups) so ASM80 listings match the
66
+ * merged image.
67
+ */
68
+ export function syncLoweredAsmInstructionBytesFromFinalBytes(program, finalBytes, env) {
69
+ for (const block of program.blocks) {
70
+ if (block.kind !== 'placed')
71
+ continue;
72
+ let offset = 0;
73
+ const origin = block.origin;
74
+ for (const item of block.items) {
75
+ if (item.kind === 'instr' && item.bytes && item.bytes.length > 0) {
76
+ const base = origin + offset;
77
+ for (let i = 0; i < item.bytes.length; i++) {
78
+ const b = finalBytes.get(base + i);
79
+ if (b !== undefined)
80
+ item.bytes[i] = b;
81
+ }
82
+ }
83
+ offset += loweredAsmItemEmittedSize(item, env);
84
+ }
85
+ }
86
+ }
87
+ function emitLoweredAsmItemBytes(item, ctx, bytes, origin, offsetRef, maxAddressRef) {
88
+ const updateMax = (addr) => {
89
+ if (addr > maxAddressRef.current)
90
+ maxAddressRef.current = addr;
91
+ };
92
+ const emitByte = (value) => {
93
+ const offset = offsetRef.current;
94
+ bytes.set(offset, toByte(value));
95
+ updateMax(origin + offset);
96
+ offsetRef.current++;
97
+ };
98
+ const emitWord = (value) => {
99
+ const v = toWord(value);
100
+ emitByte(v & 0xff);
101
+ emitByte((v >> 8) & 0xff);
102
+ };
103
+ switch (item.kind) {
104
+ case 'label':
105
+ case 'const':
106
+ case 'comment':
107
+ return;
108
+ case 'db':
109
+ for (const value of item.values) {
110
+ const v = evalLoweredImmExpr(value, ctx.env);
111
+ if (v === undefined) {
112
+ ctx.diag(ctx.diagnostics, ctx.primaryFile, 'Failed to evaluate lowered byte value.');
113
+ emitByte(0);
114
+ }
115
+ else {
116
+ emitByte(v);
117
+ }
118
+ }
119
+ return;
120
+ case 'dw':
121
+ for (const value of item.values) {
122
+ const v = evalLoweredImmExpr(value, ctx.env);
123
+ if (v === undefined) {
124
+ ctx.diag(ctx.diagnostics, ctx.primaryFile, 'Failed to evaluate lowered word value.');
125
+ emitWord(0);
126
+ }
127
+ else {
128
+ emitWord(v);
129
+ }
130
+ }
131
+ return;
132
+ case 'ds': {
133
+ const size = evalLoweredImmExpr(item.size, ctx.env);
134
+ if (size === undefined || size < 0) {
135
+ ctx.diag(ctx.diagnostics, ctx.primaryFile, 'Failed to evaluate lowered reserve size.');
136
+ return;
137
+ }
138
+ if (item.fill === undefined) {
139
+ offsetRef.current += size;
140
+ return;
141
+ }
142
+ const fillValue = evalLoweredImmExpr(item.fill, ctx.env) ?? 0;
143
+ for (let i = 0; i < size; i++)
144
+ emitByte(fillValue);
145
+ return;
146
+ }
147
+ case 'instr': {
148
+ if (!item.bytes) {
149
+ ctx.diag(ctx.diagnostics, ctx.primaryFile, `Lowered instruction missing encoded bytes.`);
150
+ return;
151
+ }
152
+ for (const b of item.bytes)
153
+ emitByte(b);
154
+ return;
155
+ }
156
+ }
157
+ }
158
+ export function emitLoweredAsmProgramBytes(program, ctx) {
159
+ const codeBytes = new Map();
160
+ const dataBytes = new Map();
161
+ const blockSizesByKey = new Map();
162
+ const maxAddressRef = { current: -1 };
163
+ for (const block of program.blocks) {
164
+ if (block.kind !== 'placed')
165
+ continue;
166
+ const placement = block.placement ?? 'code';
167
+ const key = blockPlacementKey(placement);
168
+ const target = placement === 'code'
169
+ ? codeBytes
170
+ : placement === 'data'
171
+ ? dataBytes
172
+ : new Map();
173
+ const offsetRef = { current: 0 };
174
+ for (const item of block.items) {
175
+ emitLoweredAsmItemBytes(item, ctx, target, block.origin, offsetRef, maxAddressRef);
176
+ }
177
+ blockSizesByKey.set(key, offsetRef.current);
178
+ }
179
+ return {
180
+ codeBytes,
181
+ dataBytes,
182
+ blockSizesByKey,
183
+ maxAddress: maxAddressRef.current,
184
+ };
185
+ }
@@ -0,0 +1,13 @@
1
+ import type { Diagnostic } from '../diagnosticTypes.js';
2
+ import type { LoweredAsmProgram, LoweredAsmStream } from './loweredAsmTypes.js';
3
+ type LoweredAsmPlacementContext = {
4
+ diagnostics: Diagnostic[];
5
+ diag: (diagnostics: Diagnostic[], file: string, message: string) => void;
6
+ primaryFile: string;
7
+ baseAddresses: {
8
+ codeBase: number;
9
+ dataBase: number;
10
+ };
11
+ };
12
+ export declare function placeLoweredAsmStream(stream: LoweredAsmStream, ctx: LoweredAsmPlacementContext): LoweredAsmProgram;
13
+ export {};
@@ -0,0 +1,86 @@
1
+ import { evalBinaryImmOp, evalUnaryImmOp } from './immMath.js';
2
+ import { loweredItemSize } from './loweredItemSize.js';
3
+ function baseOriginForPlacement(placement, baseAddresses) {
4
+ switch (placement) {
5
+ case 'code':
6
+ return baseAddresses.codeBase;
7
+ case 'data':
8
+ return baseAddresses.dataBase;
9
+ }
10
+ }
11
+ function resolveBlockOrigin(block, ctx) {
12
+ return baseOriginForPlacement(block.placement, ctx.baseAddresses);
13
+ }
14
+ export function placeLoweredAsmStream(stream, ctx) {
15
+ const blocks = [];
16
+ for (const block of stream.blocks) {
17
+ blocks.push({
18
+ kind: 'placed',
19
+ origin: resolveBlockOrigin(block, ctx),
20
+ placement: block.placement,
21
+ items: block.items,
22
+ });
23
+ }
24
+ resolvePlacedLoweredDataSymbols(blocks);
25
+ return { blocks };
26
+ }
27
+ function evalPlacedConst(expr, values) {
28
+ switch (expr.kind) {
29
+ case 'literal':
30
+ return expr.value;
31
+ case 'symbol': {
32
+ const base = values.get(expr.name.toLowerCase());
33
+ return base === undefined ? undefined : base + expr.addend;
34
+ }
35
+ case 'unary': {
36
+ const value = evalPlacedConst(expr.expr, values);
37
+ if (value === undefined)
38
+ return undefined;
39
+ return evalUnaryImmOp(expr.op, value);
40
+ }
41
+ case 'binary': {
42
+ const left = evalPlacedConst(expr.left, values);
43
+ const right = evalPlacedConst(expr.right, values);
44
+ if (left === undefined || right === undefined)
45
+ return undefined;
46
+ return evalBinaryImmOp(expr.op, left, right);
47
+ }
48
+ case 'opaque':
49
+ return undefined;
50
+ }
51
+ }
52
+ function resolvePlacedExpr(expr, values) {
53
+ const resolved = evalPlacedConst(expr, values);
54
+ return resolved === undefined ? expr : { kind: 'literal', value: resolved };
55
+ }
56
+ function resolvePlacedLoweredDataSymbols(blocks) {
57
+ const values = new Map();
58
+ for (const block of blocks) {
59
+ let offset = 0;
60
+ for (const item of block.items) {
61
+ if (item.kind === 'label')
62
+ values.set(item.name.toLowerCase(), block.origin + offset);
63
+ if (item.kind === 'const') {
64
+ const resolved = evalPlacedConst(item.value, values);
65
+ if (resolved !== undefined)
66
+ values.set(item.name.toLowerCase(), resolved);
67
+ }
68
+ offset += loweredItemSize(item);
69
+ }
70
+ }
71
+ for (const block of blocks) {
72
+ for (const item of block.items) {
73
+ if (item.kind === 'db' || item.kind === 'dw') {
74
+ item.values = item.values.map((value) => resolvePlacedExpr(value, values));
75
+ }
76
+ else if (item.kind === 'ds') {
77
+ item.size = resolvePlacedExpr(item.size, values);
78
+ if (item.fill)
79
+ item.fill = resolvePlacedExpr(item.fill, values);
80
+ }
81
+ else if (item.kind === 'const') {
82
+ item.value = resolvePlacedExpr(item.value, values);
83
+ }
84
+ }
85
+ }
86
+ }
@@ -0,0 +1,27 @@
1
+ import type { SourceSpan, ImmExprNode, AsmOperandNode, TypeExprNode } from '../frontend/ast.js';
2
+ import type { PlacementKind } from './loweringTypes.js';
3
+ import type { LoweredAsmItem, LoweredAsmStream, LoweredAsmStreamBlock, LoweredImmExpr, LoweredOperand } from './loweredAsmTypes.js';
4
+ type LoweredAsmStreamRecordingContext = {
5
+ activePlacementRef: {
6
+ current: PlacementKind;
7
+ };
8
+ loweredAsmBlocksByKey: Map<string, LoweredAsmStreamBlock>;
9
+ loweredAsmStream: LoweredAsmStream;
10
+ sourceLineComments?: Map<string, Map<number, string>>;
11
+ sourceTexts?: Map<string, string>;
12
+ evalImmNoDiag: (expr: ImmExprNode) => number | undefined;
13
+ symbolicTargetFromExpr: (expr: ImmExprNode) => {
14
+ baseLower: string;
15
+ addend: number;
16
+ } | undefined;
17
+ formatImmExprForAsm: (expr: ImmExprNode) => string;
18
+ typeDisplay: (typeExpr: TypeExprNode) => string;
19
+ };
20
+ export declare function createLoweredAsmStreamRecordingHelpers(ctx: LoweredAsmStreamRecordingContext): {
21
+ flushTrailingUserComments: () => void;
22
+ getLoweredAsmBlock: () => LoweredAsmStreamBlock;
23
+ lowerImmExprForLoweredAsm: (expr: ImmExprNode) => LoweredImmExpr;
24
+ lowerOperandForLoweredAsm: (operand: AsmOperandNode) => LoweredOperand;
25
+ recordLoweredAsmItem: (item: LoweredAsmItem, span?: SourceSpan) => void;
26
+ };
27
+ export {};
@@ -0,0 +1,215 @@
1
+ export function createLoweredAsmStreamRecordingHelpers(ctx) {
2
+ const { activePlacementRef, loweredAsmBlocksByKey, loweredAsmStream, sourceLineComments, sourceTexts, evalImmNoDiag, symbolicTargetFromExpr, formatImmExprForAsm, typeDisplay, } = ctx;
3
+ const getLoweredAsmBlock = () => {
4
+ const placement = activePlacementRef.current;
5
+ const key = `base:${placement}`;
6
+ let block = loweredAsmBlocksByKey.get(key);
7
+ if (!block) {
8
+ block = {
9
+ kind: 'base',
10
+ placement,
11
+ items: [],
12
+ };
13
+ loweredAsmBlocksByKey.set(key, block);
14
+ loweredAsmStream.blocks.push(block);
15
+ }
16
+ return block;
17
+ };
18
+ const commentByFileLine = new Map();
19
+ const pendingUserComments = new Map();
20
+ const emittedUserCommentLines = new Set();
21
+ const lastBlockByFile = new Map();
22
+ const registerUserCommentLines = (file, lineMap) => {
23
+ if (lineMap.size === 0)
24
+ return;
25
+ commentByFileLine.set(file, lineMap);
26
+ pendingUserComments.set(file, {
27
+ lines: [...lineMap.keys()].sort((a, b) => a - b),
28
+ texts: lineMap,
29
+ index: 0,
30
+ });
31
+ };
32
+ if (sourceLineComments) {
33
+ for (const [file, lineMap] of sourceLineComments) {
34
+ registerUserCommentLines(file, lineMap);
35
+ }
36
+ }
37
+ else if (sourceTexts) {
38
+ for (const [file, text] of sourceTexts) {
39
+ const lines = text.split(/\r?\n/);
40
+ const lineMap = new Map();
41
+ for (let i = 0; i < lines.length; i++) {
42
+ const line = lines[i] ?? '';
43
+ const semi = line.indexOf(';');
44
+ if (semi < 0)
45
+ continue;
46
+ const commentText = line.slice(semi + 1).trim();
47
+ if (!commentText)
48
+ continue;
49
+ lineMap.set(i + 1, commentText);
50
+ }
51
+ registerUserCommentLines(file, lineMap);
52
+ }
53
+ }
54
+ const emitPendingUserComments = (span) => {
55
+ if (!span)
56
+ return;
57
+ const pending = pendingUserComments.get(span.file);
58
+ if (!pending)
59
+ return;
60
+ while (pending.index < pending.lines.length) {
61
+ const line = pending.lines[pending.index];
62
+ if (line > span.start.line)
63
+ break;
64
+ pending.index += 1;
65
+ const key = `${span.file}:${line}`;
66
+ if (emittedUserCommentLines.has(key))
67
+ continue;
68
+ const text = pending.texts.get(line);
69
+ if (!text)
70
+ continue;
71
+ emittedUserCommentLines.add(key);
72
+ getLoweredAsmBlock().items.push({ kind: 'comment', text, origin: 'user' });
73
+ }
74
+ };
75
+ const recordLoweredAsmItem = (item, span) => {
76
+ if (item.kind !== 'comment' || item.origin !== 'user') {
77
+ emitPendingUserComments(span);
78
+ }
79
+ if (span) {
80
+ lastBlockByFile.set(span.file, getLoweredAsmBlock());
81
+ }
82
+ getLoweredAsmBlock().items.push(item);
83
+ };
84
+ const flushTrailingUserComments = () => {
85
+ for (const [file, pending] of pendingUserComments) {
86
+ if (pending.index >= pending.lines.length)
87
+ continue;
88
+ const block = lastBlockByFile.get(file);
89
+ if (!block)
90
+ continue;
91
+ while (pending.index < pending.lines.length) {
92
+ const line = pending.lines[pending.index];
93
+ pending.index += 1;
94
+ const key = `${file}:${line}`;
95
+ if (emittedUserCommentLines.has(key))
96
+ continue;
97
+ const text = pending.texts.get(line);
98
+ if (!text)
99
+ continue;
100
+ emittedUserCommentLines.add(key);
101
+ block.items.push({ kind: 'comment', text, origin: 'user' });
102
+ }
103
+ }
104
+ };
105
+ const lowerImmExprForLoweredAsm = (expr) => {
106
+ const value = evalImmNoDiag(expr);
107
+ if (value !== undefined)
108
+ return { kind: 'literal', value };
109
+ if (expr.kind !== 'ImmName') {
110
+ const symbolic = symbolicTargetFromExpr(expr);
111
+ if (symbolic) {
112
+ return { kind: 'symbol', name: symbolic.baseLower, addend: symbolic.addend };
113
+ }
114
+ }
115
+ switch (expr.kind) {
116
+ case 'ImmLiteral':
117
+ return { kind: 'literal', value: expr.value };
118
+ case 'ImmName':
119
+ return { kind: 'symbol', name: expr.name, addend: 0 };
120
+ case 'ImmUnary':
121
+ return {
122
+ kind: 'unary',
123
+ op: expr.op,
124
+ expr: lowerImmExprForLoweredAsm(expr.expr),
125
+ };
126
+ case 'ImmBinary':
127
+ return {
128
+ kind: 'binary',
129
+ op: expr.op,
130
+ left: lowerImmExprForLoweredAsm(expr.left),
131
+ right: lowerImmExprForLoweredAsm(expr.right),
132
+ };
133
+ default:
134
+ return { kind: 'opaque', text: formatImmExprForAsm(expr) };
135
+ }
136
+ };
137
+ const lowerEaExprForLoweredAsm = (expr) => {
138
+ switch (expr.kind) {
139
+ case 'EaName':
140
+ return { kind: 'name', name: expr.name };
141
+ case 'EaImm':
142
+ return { kind: 'imm', expr: lowerImmExprForLoweredAsm(expr.expr) };
143
+ case 'EaLayoutCast':
144
+ return {
145
+ kind: 'layoutCast',
146
+ typeName: typeDisplay(expr.typeExpr),
147
+ base: lowerEaExprForLoweredAsm(expr.base),
148
+ };
149
+ case 'EaField':
150
+ return { kind: 'field', base: lowerEaExprForLoweredAsm(expr.base), field: expr.field };
151
+ case 'EaAdd':
152
+ return {
153
+ kind: 'add',
154
+ base: lowerEaExprForLoweredAsm(expr.base),
155
+ offset: lowerImmExprForLoweredAsm(expr.offset),
156
+ };
157
+ case 'EaSub':
158
+ return {
159
+ kind: 'sub',
160
+ base: lowerEaExprForLoweredAsm(expr.base),
161
+ offset: lowerImmExprForLoweredAsm(expr.offset),
162
+ };
163
+ case 'EaIndex': {
164
+ const lowerIndexExpr = (index) => {
165
+ switch (index.kind) {
166
+ case 'IndexImm':
167
+ return { kind: 'imm', value: lowerImmExprForLoweredAsm(index.value) };
168
+ case 'IndexReg8':
169
+ return { kind: 'reg8', reg: index.reg };
170
+ case 'IndexReg16':
171
+ return { kind: 'reg16', reg: index.reg };
172
+ case 'IndexMemHL':
173
+ return { kind: 'memHL' };
174
+ case 'IndexMemIxIy':
175
+ return {
176
+ kind: 'memIxIy',
177
+ base: index.base,
178
+ ...(index.disp ? { disp: lowerImmExprForLoweredAsm(index.disp) } : {}),
179
+ };
180
+ case 'IndexEa':
181
+ return { kind: 'ea', expr: lowerEaExprForLoweredAsm(index.expr) };
182
+ }
183
+ };
184
+ return {
185
+ kind: 'index',
186
+ base: lowerEaExprForLoweredAsm(expr.base),
187
+ index: lowerIndexExpr(expr.index),
188
+ };
189
+ }
190
+ }
191
+ };
192
+ const lowerOperandForLoweredAsm = (operand) => {
193
+ switch (operand.kind) {
194
+ case 'Reg':
195
+ return { kind: 'reg', name: operand.name };
196
+ case 'Imm':
197
+ return { kind: 'imm', expr: lowerImmExprForLoweredAsm(operand.expr) };
198
+ case 'Ea':
199
+ return { kind: 'ea', expr: lowerEaExprForLoweredAsm(operand.expr) };
200
+ case 'Mem':
201
+ return { kind: 'mem', expr: lowerEaExprForLoweredAsm(operand.expr) };
202
+ case 'PortImm8':
203
+ return { kind: 'portImm8', expr: lowerImmExprForLoweredAsm(operand.expr) };
204
+ case 'PortC':
205
+ return { kind: 'portC' };
206
+ }
207
+ };
208
+ return {
209
+ flushTrailingUserComments,
210
+ getLoweredAsmBlock,
211
+ lowerImmExprForLoweredAsm,
212
+ lowerOperandForLoweredAsm,
213
+ recordLoweredAsmItem,
214
+ };
215
+ }