mirror of
https://github.com/irmen/prog8.git
synced 2025-06-15 18:23:35 +00:00
Compare commits
517 Commits
Author | SHA1 | Date | |
---|---|---|---|
c86c0912f8 | |||
268b0c9365 | |||
099fe280ba | |||
f786f60e9c | |||
f40e1eb1f2 | |||
8b9da65357 | |||
2cbbe0d48a | |||
b6e1fb3ba8 | |||
bdccffbb8e | |||
5a85474712 | |||
f50899c6fa | |||
4daa909f32 | |||
4555edf369 | |||
529ea5bf58 | |||
fe011de934 | |||
0653d430a7 | |||
a587f6e9a0 | |||
3850e1dbb5 | |||
91cde072e0 | |||
2ca4aed566 | |||
5071da6784 | |||
4c1e2f3110 | |||
2727a4dcb3 | |||
126d4c69e6 | |||
7657edcb7d | |||
580e786952 | |||
c0ae35b3a3 | |||
c3dc74788a | |||
379d241a0d | |||
1f49e8fe75 | |||
d70cfbb661 | |||
5482ac0302 | |||
131d5ceb4f | |||
512ddd1694 | |||
14a213bff9 | |||
d586846bc5 | |||
ef4efcb112 | |||
b01555d75e | |||
3804fba0f1 | |||
f93b7e3303 | |||
73baaeff1f | |||
7c79cdbd2f | |||
8ea032ed66 | |||
e7a0cb636c | |||
02f3f5d0f5 | |||
1e9bbd662b | |||
8644a4ae91 | |||
1e85f7812f | |||
80d88b3c61 | |||
d2827a7431 | |||
28c721fa7d | |||
8f799567cf | |||
9e8cc8b54d | |||
cc59069876 | |||
697d54e10a | |||
1679ca79b4 | |||
124ec77b58 | |||
3675d7961b | |||
f8aaa2d13c | |||
b7afda781a | |||
535ec13072 | |||
26d0a174db | |||
b2e821755c | |||
2e303041c1 | |||
96bed8f57f | |||
86d4a4309f | |||
1a1ab0dac6 | |||
ba8c3d14f7 | |||
617ea15c3a | |||
ef192a5778 | |||
565973c520 | |||
25b1043572 | |||
1ebfff7c7b | |||
8341f9c066 | |||
28cac291de | |||
8fa14a10e2 | |||
55dbd095ed | |||
31ad8bdd8d | |||
181f3e9eb1 | |||
50c3d809dc | |||
58f696d00a | |||
f603c543d3 | |||
6aaa0f928e | |||
feb8aa435e | |||
310e8f15cd | |||
da03941582 | |||
dcbb36a3bd | |||
53558f5c1d | |||
189399d5f8 | |||
5406a992f5 | |||
bc9683cc54 | |||
2eed75f602 | |||
d58f9f56c4 | |||
2e35f3c3a3 | |||
5c6bd9c091 | |||
857d2eefca | |||
90f1e7fd6a | |||
18e37accf9 | |||
cc53d698bf | |||
cb86206698 | |||
d77b1944fb | |||
a58cb43c4a | |||
88574c87c4 | |||
3a7a7091c0 | |||
906b137a7c | |||
42e2c5f605 | |||
cc13a51493 | |||
f569ce6141 | |||
4958463e75 | |||
2360625927 | |||
8badc40883 | |||
844c97930f | |||
5c09dc10ae | |||
9fd9e9ab5f | |||
35c477b5a6 | |||
ae0cadb383 | |||
984230e8fa | |||
a874aec6a1 | |||
ea1daa97d3 | |||
fb0d9b46b0 | |||
9da70bdf05 | |||
d640cfbe13 | |||
51a05ec4b7 | |||
1f5706bbeb | |||
25c9b2fea4 | |||
154f9b300f | |||
d78ce77536 | |||
b4fb43bc80 | |||
e0e01f794e | |||
08865dbb4e | |||
b9ad7e0e55 | |||
07158a6f1a | |||
957c42bc1d | |||
f784da2da6 | |||
c080fbe59a | |||
d70b8303b1 | |||
1d38c3582a | |||
9438e996d7 | |||
3b4a5e27f7 | |||
648d9fc269 | |||
54fccec7d7 | |||
4f9693055e | |||
555c50ee10 | |||
bf98ceca2c | |||
573cecb087 | |||
1b528491c2 | |||
4bdabe1961 | |||
a3fa527378 | |||
84f5ffa426 | |||
25d2b42283 | |||
300d1a871c | |||
2fcb83a39f | |||
3ba1d00a7c | |||
64164c1c72 | |||
3ee6058524 | |||
93a0a41e73 | |||
e7ab7b6d7a | |||
7d4dc3c063 | |||
a50400b7d1 | |||
f89f1a84d0 | |||
688dce6145 | |||
b88f550c5b | |||
9864abd393 | |||
c702c4a6df | |||
77e376f6bf | |||
491e5dbcfb | |||
a5c7393561 | |||
7fd3e9bb7d | |||
459e9f8f3b | |||
5b1143bcb3 | |||
fddd390d31 | |||
e514eeba17 | |||
c11a52b278 | |||
85e87dfe2e | |||
cb47e2c149 | |||
0fc9aa6b2d | |||
155896c4c7 | |||
178e60bba0 | |||
9f84aa5fb2 | |||
66fc109ce5 | |||
a231872821 | |||
7cfb33a448 | |||
3b798097b9 | |||
6fb05bdefc | |||
64ea72ed4d | |||
89425088ce | |||
925b9d845d | |||
ad074076c2 | |||
a2194c43a6 | |||
4b23b1dc86 | |||
9005c7994a | |||
4a47e15b1c | |||
09cbdf410a | |||
df6a43c7f0 | |||
4ce130dc8b | |||
94d76aa82c | |||
73609636c5 | |||
66b06d6c40 | |||
eeeb8d81f4 | |||
6f727aff88 | |||
518e5a30c2 | |||
bbba4b3d60 | |||
967adb9a87 | |||
040a6c62de | |||
483d193ced | |||
62458216c9 | |||
76b05cb5fd | |||
570b574b93 | |||
a82f211f9a | |||
504c80cddf | |||
4b4af9b527 | |||
28b383f888 | |||
40ce7725a1 | |||
1f2d46628e | |||
c9535049c8 | |||
9317cf8a35 | |||
1cd754f05d | |||
97b8cb748d | |||
84d9040b57 | |||
fdd18c615c | |||
c14f6cfc2b | |||
326eab3dd1 | |||
6da1f7eb4c | |||
1e82483152 | |||
6e2fd41a8b | |||
9927af1095 | |||
7585b6ef6f | |||
a6159702da | |||
0247fb0d84 | |||
6de760885f | |||
9851d14fb9 | |||
d5fc69d3e4 | |||
a40d120f2a | |||
fcdd9414d9 | |||
272a1001a8 | |||
2a52241f1c | |||
d8f1822c12 | |||
ce7d094adb | |||
a0cf1889a3 | |||
38ef394e15 | |||
abbf7c7cb0 | |||
ca5f7ae32f | |||
cbc4b75e50 | |||
65ddcf91d0 | |||
5280e1b449 | |||
b6ffb81909 | |||
e9edffa9f0 | |||
0dd1c17ff4 | |||
aef211e5f3 | |||
66829203d8 | |||
7a0eaf3148 | |||
fa5479ee5f | |||
03412cacba | |||
01a38a0b11 | |||
f43c14bd78 | |||
fb23452383 | |||
ab7dde1450 | |||
8d9bc2f5ff | |||
7651ccc84e | |||
1a6b95b388 | |||
78ec1e7512 | |||
7e38d26c33 | |||
ed09dd4e9e | |||
5731b79554 | |||
eaa22a9d13 | |||
b2bdfe8482 | |||
fea531be9a | |||
7c69d38588 | |||
a088ee56b0 | |||
ae669af904 | |||
d1ddf05e38 | |||
51279a98b3 | |||
bf33a4f82d | |||
fff0d741c3 | |||
e83d0ee820 | |||
09f3eecf56 | |||
2bd4326ff6 | |||
c13168b60c | |||
ea3871d0c4 | |||
70a2b11271 | |||
3cf39e072e | |||
413b86cc4a | |||
a6107fcfdf | |||
a064ade1e0 | |||
df35aa7942 | |||
cd49c5f88d | |||
1541ad2160 | |||
c78b7b1a24 | |||
9c7a645e18 | |||
4acf38031a | |||
4cd7271e30 | |||
3f630ab1b0 | |||
04cb684fd4 | |||
4c843571ea | |||
1326498802 | |||
b7ebd8c4a6 | |||
24e0a69480 | |||
4bcb2bdede | |||
d27f3eb8a4 | |||
d3e4481112 | |||
1d1d6b3d98 | |||
90b8a22a71 | |||
8dbfb8ab76 | |||
e29ff1c848 | |||
585f6ffc9b | |||
46b94c17d6 | |||
7af8007447 | |||
16a2b2f566 | |||
ea2a90c3c5 | |||
5cda750e5e | |||
4e143d45c8 | |||
4c50980d81 | |||
2954f5f04d | |||
cac4c1eb1e | |||
0b1f30d98c | |||
c7b1e8d772 | |||
a4f7512d44 | |||
0d3ad80659 | |||
aba1a73e28 | |||
dca31b2ca3 | |||
0cb378ca31 | |||
cf551d2cc7 | |||
ac0c8a68f6 | |||
5986dcdd2f | |||
6be6eb2227 | |||
d34015eec5 | |||
255c5bfaca | |||
01c6754928 | |||
8eaf884f69 | |||
699a2bb7ab | |||
4a2dcd20d1 | |||
4e98fb75d6 | |||
64e66e732f | |||
7aec627f6b | |||
59a2fec176 | |||
edc5a5a94f | |||
c5b7edad82 | |||
124ffac4e4 | |||
6d2a36fb2b | |||
28b43b3e1d | |||
f7feaf158d | |||
2396f707c6 | |||
d4d8e1b1ba | |||
44fec2c729 | |||
a80a6913e3 | |||
0eac04c220 | |||
29dd758302 | |||
5c45adc7f0 | |||
ad22cf08cd | |||
2c2ae64194 | |||
97c2dadd16 | |||
b36e1e3baf | |||
2da35fec17 | |||
bdeac74cfc | |||
6516d7cb15 | |||
31cf76042d | |||
c4c4dcf2b3 | |||
03145630f8 | |||
e2fcac322f | |||
beaff4d650 | |||
79af96ddde | |||
e439720c9d | |||
48d0185ea4 | |||
e2592b4e0b | |||
2967866e3d | |||
b566ea5c3f | |||
8f6eaeac2c | |||
b4facaeb3c | |||
d12b7ccc6b | |||
453e8bd0a0 | |||
9204d390ae | |||
b70ce0015c | |||
d113827753 | |||
c67f877857 | |||
0ec719e429 | |||
17f7b11148 | |||
966b017670 | |||
4c98070b3c | |||
3681d6ee1c | |||
0af17cdc33 | |||
2aae1f5e30 | |||
d18f2a7bfd | |||
9046fe8d3a | |||
78c7ee247a | |||
d5adb85e5b | |||
69f953fd9b | |||
484677b4b1 | |||
b10a8e728f | |||
25f25a8767 | |||
0c053e4a2c | |||
3f6521cc9b | |||
a074491d5b | |||
a291164953 | |||
43c55b58d2 | |||
e7298f8162 | |||
ddf990296b | |||
ead8aa7800 | |||
7a9dd1ac9b | |||
1c97c22eff | |||
bbf621a8c4 | |||
8efa89165c | |||
4f8aaf9244 | |||
a97edef380 | |||
eefae24aa3 | |||
54bffc91ae | |||
63f5ef9e14 | |||
034f27a8dd | |||
c2f6311367 | |||
6f00a48772 | |||
b3dba67405 | |||
c9a4235669 | |||
ae0d52274c | |||
8973763866 | |||
3d799ae7fe | |||
8b10115390 | |||
d2e010c439 | |||
15867ab423 | |||
22c9e99fa3 | |||
ee262f6aad | |||
af64af2397 | |||
1feead2260 | |||
d3dcd24b4d | |||
fd1e6796ef | |||
3ea0f0cbaa | |||
f3e3311598 | |||
0dc50a93a4 | |||
fda8e61be4 | |||
ac1d4b4a7a | |||
c719e274d5 | |||
e4990f8ec5 | |||
62afd3342e | |||
6e8a89e6f1 | |||
aa2437cfb8 | |||
4a710ecdfc | |||
3ef5bdfeda | |||
7915dda35f | |||
9120e16683 | |||
a1ebc7090d | |||
054b4636e0 | |||
e3e7b060b7 | |||
5ac9c75521 | |||
07710e0995 | |||
d6a67f5f2b | |||
2675623aea | |||
94263c43d0 | |||
d8ec03874f | |||
a7247f5b8b | |||
4d37581694 | |||
5d7ddebcad | |||
53df0eb707 | |||
8babad9c7c | |||
8db7aa07bd | |||
42f4b06ac8 | |||
f4b50368ba | |||
db80417bd7 | |||
7a6f2ecc8c | |||
f5d556a7f9 | |||
2aae46d632 | |||
19ebc6d6b3 | |||
f88c29e083 | |||
6ed9899dc7 | |||
9de7698a5c | |||
112d2d6058 | |||
ddb8346711 | |||
8dd3faf395 | |||
35f3e8708b | |||
cfe3fcc9e7 | |||
66a6659a6e | |||
88ae3daa42 | |||
08b8fe01ab | |||
731132d4b3 | |||
98acff802f | |||
5f11f485a2 | |||
34f3169dda | |||
a3ef8f814b | |||
385dd6fc23 | |||
9af4168ae2 | |||
a5e0e31b74 | |||
b385dc8c26 | |||
92c012b55a | |||
641f6c05d8 | |||
788f6b44a6 | |||
63a4525f06 | |||
3e34a3ef72 | |||
0c5e8ca199 | |||
ff23fb0086 | |||
56f41d5e34 | |||
4700a239b9 | |||
bd5abfb969 | |||
b93fa75377 | |||
681ce9c60c | |||
dd0f0fe415 | |||
119040fc50 | |||
551e5688da | |||
56c1035581 | |||
ba1e907c79 | |||
2a3a27c56d | |||
647af34f5b | |||
993be6394e | |||
9a27505315 | |||
2e37f5dee3 | |||
03e486c082 | |||
edc83305a4 | |||
66e7c51064 | |||
60244aaf16 | |||
443391c700 | |||
47dbafacd4 | |||
5b6811d073 | |||
7516116bb7 | |||
e6014ea4dd | |||
362abfe284 | |||
ad4880997a | |||
592becc126 | |||
c38765301e | |||
5f27426f59 | |||
d924f8bff8 | |||
d14c61b160 |
22
.github/workflows/all-ci.yml
vendored
22
.github/workflows/all-ci.yml
vendored
@ -18,30 +18,24 @@ jobs:
|
|||||||
git clone --depth=1 https://github.com/irmen/64tass
|
git clone --depth=1 https://github.com/irmen/64tass
|
||||||
cd 64tass
|
cd 64tass
|
||||||
make -j4
|
make -j4
|
||||||
sudo make install
|
sudo make install
|
||||||
|
|
||||||
- name: Set up JDK 11
|
- name: Set up JDK 11
|
||||||
uses: actions/setup-java@v4
|
uses: actions/setup-java@v4
|
||||||
with:
|
with:
|
||||||
java-version: 11
|
java-version: 11
|
||||||
distribution: adopt
|
distribution: temurin
|
||||||
|
|
||||||
- name: Build and test with Gradle
|
- name: Build and test with Gradle
|
||||||
run: ./gradlew build shadowJar --no-daemon
|
run: |
|
||||||
|
./gradlew build shadowJar --no-daemon
|
||||||
|
sha256sum -b compiler/build/libs/*-all.jar > compiler/build/libs/hash.txt
|
||||||
|
|
||||||
- name: Create compiler shadowJar artifact
|
- name: Create compiler shadowJar artifact
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v4
|
||||||
with:
|
with:
|
||||||
name: prog8-compiler-jar-zipped
|
name: prog8-compiler-jar-zipped
|
||||||
path: compiler/build/libs/*-all.jar
|
path: |
|
||||||
|
compiler/build/libs/*-all.jar
|
||||||
|
compiler/build/libs/hash.txt
|
||||||
|
|
||||||
- uses: MCJack123/ghaction-generate-release-hashes@v4
|
|
||||||
if: "github.event_name == 'release'"
|
|
||||||
with:
|
|
||||||
hash-type: sha256
|
|
||||||
file-name: hashes.txt
|
|
||||||
- uses: actions/upload-artifact@v4
|
|
||||||
if: "github.event_name == 'release'"
|
|
||||||
with:
|
|
||||||
name: Artifact Hashes
|
|
||||||
path: hashes.txt
|
|
||||||
|
7
.gitignore
vendored
7
.gitignore
vendored
@ -1,8 +1,12 @@
|
|||||||
.idea/workspace.xml
|
.idea/workspace.xml
|
||||||
.idea/discord.xml
|
.idea/discord.xml
|
||||||
|
.idea/developer-tools.xml
|
||||||
|
.idea/usage.statistics.xml
|
||||||
|
.idea/shelf/
|
||||||
build/
|
build/
|
||||||
dist/
|
dist/
|
||||||
output/
|
output/
|
||||||
|
out/
|
||||||
.*cache/
|
.*cache/
|
||||||
*.directory
|
*.directory
|
||||||
*.prg
|
*.prg
|
||||||
@ -11,7 +15,6 @@ output/
|
|||||||
*.vm.txt
|
*.vm.txt
|
||||||
*.vice-mon-list
|
*.vice-mon-list
|
||||||
docs/build
|
docs/build
|
||||||
out/
|
|
||||||
parser/**/*.interp
|
parser/**/*.interp
|
||||||
parser/**/*.tokens
|
parser/**/*.tokens
|
||||||
parser/**/*.java
|
parser/**/*.java
|
||||||
@ -22,6 +25,7 @@ compiler/src/prog8/buildversion/*
|
|||||||
.eggs/
|
.eggs/
|
||||||
/MANIFEST
|
/MANIFEST
|
||||||
.tox/
|
.tox/
|
||||||
|
.kotlin/
|
||||||
__pycache__/
|
__pycache__/
|
||||||
parser.out
|
parser.out
|
||||||
parsetab.py
|
parsetab.py
|
||||||
@ -30,7 +34,6 @@ parsetab.py
|
|||||||
compiler/lib/
|
compiler/lib/
|
||||||
|
|
||||||
.gradle
|
.gradle
|
||||||
/prog8compiler.jar
|
|
||||||
sd*.img
|
sd*.img
|
||||||
*.d64
|
*.d64
|
||||||
|
|
||||||
|
12
.idea/kotlinc.xml
generated
12
.idea/kotlinc.xml
generated
@ -1,9 +1,19 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<project version="4">
|
<project version="4">
|
||||||
|
<component name="Kotlin2JsCompilerArguments">
|
||||||
|
<option name="moduleKind" value="plain" />
|
||||||
|
</component>
|
||||||
<component name="Kotlin2JvmCompilerArguments">
|
<component name="Kotlin2JvmCompilerArguments">
|
||||||
<option name="jvmTarget" value="11" />
|
<option name="jvmTarget" value="11" />
|
||||||
</component>
|
</component>
|
||||||
|
<component name="KotlinCommonCompilerArguments">
|
||||||
|
<option name="apiVersion" value="2.1" />
|
||||||
|
<option name="languageVersion" value="2.1" />
|
||||||
|
</component>
|
||||||
|
<component name="KotlinCompilerSettings">
|
||||||
|
<option name="additionalArguments" value="-Xwhen-guards" />
|
||||||
|
</component>
|
||||||
<component name="KotlinJpsPluginSettings">
|
<component name="KotlinJpsPluginSettings">
|
||||||
<option name="version" value="1.9.20" />
|
<option name="version" value="2.1.0" />
|
||||||
</component>
|
</component>
|
||||||
</project>
|
</project>
|
20
.idea/libraries/KotlinJavaRuntime.xml
generated
20
.idea/libraries/KotlinJavaRuntime.xml
generated
@ -1,23 +1,23 @@
|
|||||||
<component name="libraryTable">
|
<component name="libraryTable">
|
||||||
<library name="KotlinJavaRuntime" type="repository">
|
<library name="KotlinJavaRuntime" type="repository">
|
||||||
<properties maven-id="org.jetbrains.kotlin:kotlin-stdlib-jdk8:1.9.22" />
|
<properties maven-id="org.jetbrains.kotlin:kotlin-stdlib-jdk8:2.1.0" />
|
||||||
<CLASSES>
|
<CLASSES>
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/1.9.22/kotlin-stdlib-jdk8-1.9.22.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/2.1.0/kotlin-stdlib-jdk8-2.1.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.22/kotlin-stdlib-1.9.22.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/2.1.0/kotlin-stdlib-2.1.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk7/1.9.22/kotlin-stdlib-jdk7-1.9.22.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk7/2.1.0/kotlin-stdlib-jdk7-2.1.0.jar!/" />
|
||||||
</CLASSES>
|
</CLASSES>
|
||||||
<JAVADOC>
|
<JAVADOC>
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/1.9.22/kotlin-stdlib-jdk8-1.9.22-javadoc.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/2.1.0/kotlin-stdlib-jdk8-2.1.0-javadoc.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.22/kotlin-stdlib-1.9.22-javadoc.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/2.1.0/kotlin-stdlib-2.1.0-javadoc.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0-javadoc.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0-javadoc.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk7/1.9.22/kotlin-stdlib-jdk7-1.9.22-javadoc.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk7/2.1.0/kotlin-stdlib-jdk7-2.1.0-javadoc.jar!/" />
|
||||||
</JAVADOC>
|
</JAVADOC>
|
||||||
<SOURCES>
|
<SOURCES>
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/1.9.22/kotlin-stdlib-jdk8-1.9.22-sources.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/2.1.0/kotlin-stdlib-jdk8-2.1.0-sources.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.22/kotlin-stdlib-1.9.22-sources.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/2.1.0/kotlin-stdlib-2.1.0-sources.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0-sources.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0-sources.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk7/1.9.22/kotlin-stdlib-jdk7-1.9.22-sources.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk7/2.1.0/kotlin-stdlib-jdk7-2.1.0-sources.jar!/" />
|
||||||
</SOURCES>
|
</SOURCES>
|
||||||
</library>
|
</library>
|
||||||
</component>
|
</component>
|
6
.idea/libraries/antlr_antlr4.xml
generated
6
.idea/libraries/antlr_antlr4.xml
generated
@ -1,13 +1,13 @@
|
|||||||
<component name="libraryTable">
|
<component name="libraryTable">
|
||||||
<library name="antlr.antlr4" type="repository">
|
<library name="antlr.antlr4" type="repository">
|
||||||
<properties maven-id="org.antlr:antlr4:4.13.1">
|
<properties maven-id="org.antlr:antlr4:4.13.2">
|
||||||
<exclude>
|
<exclude>
|
||||||
<dependency maven-id="com.ibm.icu:icu4j" />
|
<dependency maven-id="com.ibm.icu:icu4j" />
|
||||||
</exclude>
|
</exclude>
|
||||||
</properties>
|
</properties>
|
||||||
<CLASSES>
|
<CLASSES>
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/antlr/antlr4/4.13.1/antlr4-4.13.1.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/antlr/antlr4/4.13.2/antlr4-4.13.2.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/antlr/antlr4-runtime/4.13.1/antlr4-runtime-4.13.1.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/antlr/antlr4-runtime/4.13.2/antlr4-runtime-4.13.2.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/antlr/antlr-runtime/3.5.3/antlr-runtime-3.5.3.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/antlr/antlr-runtime/3.5.3/antlr-runtime-3.5.3.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/antlr/ST4/4.3.4/ST4-4.3.4.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/antlr/ST4/4.3.4/ST4-4.3.4.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/abego/treelayout/org.abego.treelayout.core/1.0.3/org.abego.treelayout.core-1.0.3.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/abego/treelayout/org.abego.treelayout.core/1.0.3/org.abego.treelayout.core-1.0.3.jar!/" />
|
||||||
|
13
.idea/libraries/eclipse_lsp4j.xml
generated
Normal file
13
.idea/libraries/eclipse_lsp4j.xml
generated
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
<component name="libraryTable">
|
||||||
|
<library name="eclipse.lsp4j" type="repository">
|
||||||
|
<properties maven-id="org.eclipse.lsp4j:org.eclipse.lsp4j:0.23.1" />
|
||||||
|
<CLASSES>
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/eclipse/lsp4j/org.eclipse.lsp4j/0.23.1/org.eclipse.lsp4j-0.23.1.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/eclipse/lsp4j/org.eclipse.lsp4j.jsonrpc/0.23.1/org.eclipse.lsp4j.jsonrpc-0.23.1.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/com/google/code/gson/gson/2.11.0/gson-2.11.0.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/com/google/errorprone/error_prone_annotations/2.27.0/error_prone_annotations-2.27.0.jar!/" />
|
||||||
|
</CLASSES>
|
||||||
|
<JAVADOC />
|
||||||
|
<SOURCES />
|
||||||
|
</library>
|
||||||
|
</component>
|
23
.idea/libraries/io_kotest_assertions_core_jvm.xml
generated
23
.idea/libraries/io_kotest_assertions_core_jvm.xml
generated
@ -1,21 +1,18 @@
|
|||||||
<component name="libraryTable">
|
<component name="libraryTable">
|
||||||
<library name="io.kotest.assertions.core.jvm" type="repository">
|
<library name="io.kotest.assertions.core.jvm" type="repository">
|
||||||
<properties maven-id="io.kotest:kotest-assertions-core-jvm:5.8.0" />
|
<properties maven-id="io.kotest:kotest-assertions-core-jvm:5.9.1" />
|
||||||
<CLASSES>
|
<CLASSES>
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-core-jvm/5.8.0/kotest-assertions-core-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-core-jvm/5.9.1/kotest-assertions-core-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/1.8.10/kotlin-stdlib-jdk8-1.8.10.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-shared-jvm/5.9.1/kotest-assertions-shared-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.8.10/kotlin-stdlib-1.8.10.jar!/" />
|
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk7/1.8.10/kotlin-stdlib-jdk7-1.8.10.jar!/" />
|
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-shared-jvm/5.8.0/kotest-assertions-shared-jvm-5.8.0.jar!/" />
|
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/github/java-diff-utils/java-diff-utils/4.12/java-diff-utils-4.12.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/github/java-diff-utils/java-diff-utils/4.12/java-diff-utils-4.12.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/opentest4j/opentest4j/1.3.0/opentest4j-1.3.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/opentest4j/opentest4j/1.3.0/opentest4j-1.3.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-common/1.8.10/kotlin-stdlib-common-1.8.10.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.23/kotlin-stdlib-1.9.23.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-jdk8/1.7.0/kotlinx-coroutines-jdk8-1.7.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-reflect/1.8.10/kotlin-reflect-1.8.10.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-jdk8/1.8.0/kotlinx-coroutines-jdk8-1.8.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-common-jvm/5.8.0/kotest-common-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-reflect/1.9.23/kotlin-reflect-1.9.23.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-api-jvm/5.8.0/kotest-assertions-api-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-common-jvm/5.9.1/kotest-common-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-core-jvm/1.7.0/kotlinx-coroutines-core-jvm-1.7.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-api-jvm/5.9.1/kotest-assertions-api-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/23.0.0/annotations-23.0.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-core-jvm/1.8.0/kotlinx-coroutines-core-jvm-1.8.0.jar!/" />
|
||||||
</CLASSES>
|
</CLASSES>
|
||||||
<JAVADOC />
|
<JAVADOC />
|
||||||
<SOURCES />
|
<SOURCES />
|
||||||
|
23
.idea/libraries/io_kotest_framework_datatest.xml
generated
Normal file
23
.idea/libraries/io_kotest_framework_datatest.xml
generated
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
<component name="libraryTable">
|
||||||
|
<library name="io.kotest.framework.datatest" type="repository">
|
||||||
|
<properties maven-id="io.kotest:kotest-framework-datatest:5.9.1" />
|
||||||
|
<CLASSES>
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-datatest/5.9.1/kotest-framework-datatest-5.9.1.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-datatest-jvm/5.9.1/kotest-framework-datatest-jvm-5.9.1.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.23/kotlin-stdlib-1.9.23.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-core-jvm/1.8.0/kotlinx-coroutines-core-jvm-1.8.0.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-common-jvm/5.9.1/kotest-common-jvm-5.9.1.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-reflect/1.9.23/kotlin-reflect-1.9.23.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-api-jvm/5.9.1/kotest-framework-api-jvm-5.9.1.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-shared-jvm/5.9.1/kotest-assertions-shared-jvm-5.9.1.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-api-jvm/5.9.1/kotest-assertions-api-jvm-5.9.1.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-jdk8/1.8.0/kotlinx-coroutines-jdk8-1.8.0.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/io/github/java-diff-utils/java-diff-utils/4.12/java-diff-utils-4.12.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/opentest4j/opentest4j/1.3.0/opentest4j-1.3.0.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-test-jvm/1.8.0/kotlinx-coroutines-test-jvm-1.8.0.jar!/" />
|
||||||
|
</CLASSES>
|
||||||
|
<JAVADOC />
|
||||||
|
<SOURCES />
|
||||||
|
</library>
|
||||||
|
</component>
|
39
.idea/libraries/io_kotest_runner_junit5_jvm.xml
generated
39
.idea/libraries/io_kotest_runner_junit5_jvm.xml
generated
@ -1,30 +1,30 @@
|
|||||||
<component name="libraryTable">
|
<component name="libraryTable">
|
||||||
<library name="io.kotest.runner.junit5.jvm" type="repository">
|
<library name="io.kotest.runner.junit5.jvm" type="repository">
|
||||||
<properties maven-id="io.kotest:kotest-runner-junit5-jvm:5.8.0" />
|
<properties maven-id="io.kotest:kotest-runner-junit5-jvm:5.9.1" />
|
||||||
<CLASSES>
|
<CLASSES>
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-runner-junit5-jvm/5.8.0/kotest-runner-junit5-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-runner-junit5-jvm/5.9.1/kotest-runner-junit5-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-api-jvm/5.8.0/kotest-framework-api-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-api-jvm/5.9.1/kotest-framework-api-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-shared-jvm/5.8.0/kotest-assertions-shared-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-shared-jvm/5.9.1/kotest-assertions-shared-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/github/java-diff-utils/java-diff-utils/4.12/java-diff-utils-4.12.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/github/java-diff-utils/java-diff-utils/4.12/java-diff-utils-4.12.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-test-jvm/1.7.0/kotlinx-coroutines-test-jvm-1.7.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-test-jvm/1.8.0/kotlinx-coroutines-test-jvm-1.8.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-common-jvm/5.8.0/kotest-common-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-common-jvm/5.9.1/kotest-common-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-engine-jvm/5.8.0/kotest-framework-engine-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-engine-jvm/5.9.1/kotest-framework-engine-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/github/classgraph/classgraph/4.8.162/classgraph-4.8.162.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/github/classgraph/classgraph/4.8.172/classgraph-4.8.172.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/com/github/ajalt/mordant/1.2.1/mordant-1.2.1.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/com/github/ajalt/mordant/1.2.1/mordant-1.2.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/com/github/ajalt/colormath/1.2.0/colormath-1.2.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/com/github/ajalt/colormath/1.2.0/colormath-1.2.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/opentest4j/opentest4j/1.3.0/opentest4j-1.3.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/opentest4j/opentest4j/1.3.0/opentest4j-1.3.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-debug/1.7.0/kotlinx-coroutines-debug-1.7.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-debug/1.8.0/kotlinx-coroutines-debug-1.8.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/net/java/dev/jna/jna/5.9.0/jna-5.9.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/net/java/dev/jna/jna/5.9.0/jna-5.9.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/net/java/dev/jna/jna-platform/5.9.0/jna-platform-5.9.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/net/java/dev/jna/jna-platform/5.9.0/jna-platform-5.9.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/net/bytebuddy/byte-buddy/1.10.9/byte-buddy-1.10.9.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/net/bytebuddy/byte-buddy/1.10.9/byte-buddy-1.10.9.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/net/bytebuddy/byte-buddy-agent/1.10.9/byte-buddy-agent-1.10.9.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/net/bytebuddy/byte-buddy-agent/1.10.9/byte-buddy-agent-1.10.9.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-discovery-jvm/5.8.0/kotest-framework-discovery-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-discovery-jvm/5.9.1/kotest-framework-discovery-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-core-jvm/5.8.0/kotest-assertions-core-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-core-jvm/5.9.1/kotest-assertions-core-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-jdk8/1.7.0/kotlinx-coroutines-jdk8-1.7.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-jdk8/1.8.0/kotlinx-coroutines-jdk8-1.8.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-api-jvm/5.8.0/kotest-assertions-api-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-api-jvm/5.9.1/kotest-assertions-api-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-extensions-jvm/5.8.0/kotest-extensions-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-extensions-jvm/5.9.1/kotest-extensions-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-concurrency-jvm/5.8.0/kotest-framework-concurrency-jvm-5.8.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-concurrency-jvm/5.9.1/kotest-framework-concurrency-jvm-5.9.1.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-core-jvm/1.7.0/kotlinx-coroutines-core-jvm-1.7.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-core-jvm/1.8.0/kotlinx-coroutines-core-jvm-1.8.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/23.0.0/annotations-23.0.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/23.0.0/annotations-23.0.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-engine/1.8.2/junit-platform-engine-1.8.2.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-engine/1.8.2/junit-platform-engine-1.8.2.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-commons/1.8.2/junit-platform-commons-1.8.2.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-commons/1.8.2/junit-platform-commons-1.8.2.jar!/" />
|
||||||
@ -32,11 +32,8 @@
|
|||||||
<root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-suite-api/1.8.2/junit-platform-suite-api-1.8.2.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-suite-api/1.8.2/junit-platform-suite-api-1.8.2.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-launcher/1.8.2/junit-platform-launcher-1.8.2.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-launcher/1.8.2/junit-platform-launcher-1.8.2.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/junit/jupiter/junit-jupiter-api/5.8.2/junit-jupiter-api-5.8.2.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/junit/jupiter/junit-jupiter-api/5.8.2/junit-jupiter-api-5.8.2.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/1.8.10/kotlin-stdlib-jdk8-1.8.10.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.23/kotlin-stdlib-1.9.23.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.8.10/kotlin-stdlib-1.8.10.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-reflect/1.9.23/kotlin-reflect-1.9.23.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk7/1.8.10/kotlin-stdlib-jdk7-1.8.10.jar!/" />
|
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-common/1.8.10/kotlin-stdlib-common-1.8.10.jar!/" />
|
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-reflect/1.8.10/kotlin-reflect-1.8.10.jar!/" />
|
|
||||||
</CLASSES>
|
</CLASSES>
|
||||||
<JAVADOC />
|
<JAVADOC />
|
||||||
<SOURCES />
|
<SOURCES />
|
||||||
|
16
.idea/libraries/michael_bull_kotlin_result_jvm.xml
generated
16
.idea/libraries/michael_bull_kotlin_result_jvm.xml
generated
@ -1,12 +1,20 @@
|
|||||||
<component name="libraryTable">
|
<component name="libraryTable">
|
||||||
<library name="michael.bull.kotlin.result.jvm" type="repository">
|
<library name="michael.bull.kotlin.result.jvm" type="repository">
|
||||||
<properties maven-id="com.michael-bull.kotlin-result:kotlin-result-jvm:1.1.20" />
|
<properties maven-id="com.michael-bull.kotlin-result:kotlin-result-jvm:2.0.0" />
|
||||||
<CLASSES>
|
<CLASSES>
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/com/michael-bull/kotlin-result/kotlin-result-jvm/1.1.20/kotlin-result-jvm-1.1.20.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/com/michael-bull/kotlin-result/kotlin-result-jvm/2.0.0/kotlin-result-jvm-2.0.0.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.22/kotlin-stdlib-1.9.22.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.22/kotlin-stdlib-1.9.22.jar!/" />
|
||||||
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0.jar!/" />
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0.jar!/" />
|
||||||
</CLASSES>
|
</CLASSES>
|
||||||
<JAVADOC />
|
<JAVADOC>
|
||||||
<SOURCES />
|
<root url="jar://$MAVEN_REPOSITORY$/com/michael-bull/kotlin-result/kotlin-result-jvm/2.0.0/kotlin-result-jvm-2.0.0-javadoc.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.22/kotlin-stdlib-1.9.22-javadoc.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0-javadoc.jar!/" />
|
||||||
|
</JAVADOC>
|
||||||
|
<SOURCES>
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/com/michael-bull/kotlin-result/kotlin-result-jvm/2.0.0/kotlin-result-jvm-2.0.0-sources.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.9.22/kotlin-stdlib-1.9.22-sources.jar!/" />
|
||||||
|
<root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0-sources.jar!/" />
|
||||||
|
</SOURCES>
|
||||||
</library>
|
</library>
|
||||||
</component>
|
</component>
|
6
.idea/misc.xml
generated
6
.idea/misc.xml
generated
@ -4,13 +4,13 @@
|
|||||||
<option name="perGrammarGenerationSettings">
|
<option name="perGrammarGenerationSettings">
|
||||||
<list>
|
<list>
|
||||||
<PerGrammarGenerationSettings>
|
<PerGrammarGenerationSettings>
|
||||||
<option name="fileName" value="$PROJECT_DIR$/parser/antlr/Prog8ANTLR.g4" />
|
<option name="fileName" value="$PROJECT_DIR$/parser/src/main/antlr/Prog8ANTLR.g4" />
|
||||||
<option name="autoGen" value="true" />
|
<option name="autoGen" value="true" />
|
||||||
<option name="outputDir" value="$PROJECT_DIR$/parser/src/prog8/parser" />
|
<option name="outputDir" value="$PROJECT_DIR$/parser/src/prog8/parser" />
|
||||||
<option name="libDir" value="" />
|
<option name="libDir" value="" />
|
||||||
<option name="encoding" value="" />
|
<option name="encoding" value="" />
|
||||||
<option name="pkg" value="" />
|
<option name="pkg" value="" />
|
||||||
<option name="language" value="" />
|
<option name="language" value="Java" />
|
||||||
<option name="generateListener" value="false" />
|
<option name="generateListener" value="false" />
|
||||||
</PerGrammarGenerationSettings>
|
</PerGrammarGenerationSettings>
|
||||||
</list>
|
</list>
|
||||||
@ -22,7 +22,7 @@
|
|||||||
<component name="FrameworkDetectionExcludesConfiguration">
|
<component name="FrameworkDetectionExcludesConfiguration">
|
||||||
<type id="Python" />
|
<type id="Python" />
|
||||||
</component>
|
</component>
|
||||||
<component name="ProjectRootManager" version="2" languageLevel="JDK_11" default="true" project-jdk-name="openjdk-11" project-jdk-type="JavaSDK">
|
<component name="ProjectRootManager" version="2" languageLevel="JDK_17" default="true" project-jdk-name="openjdk-17" project-jdk-type="JavaSDK">
|
||||||
<output url="file://$PROJECT_DIR$/out" />
|
<output url="file://$PROJECT_DIR$/out" />
|
||||||
</component>
|
</component>
|
||||||
</project>
|
</project>
|
3
.idea/modules.xml
generated
3
.idea/modules.xml
generated
@ -2,6 +2,8 @@
|
|||||||
<project version="4">
|
<project version="4">
|
||||||
<component name="ProjectModuleManager">
|
<component name="ProjectModuleManager">
|
||||||
<modules>
|
<modules>
|
||||||
|
<module fileurl="file://$PROJECT_DIR$/beanshell/beanshell.iml" filepath="$PROJECT_DIR$/beanshell/beanshell.iml" />
|
||||||
|
<module fileurl="file://$PROJECT_DIR$/benchmark-program/benchmark-program.iml" filepath="$PROJECT_DIR$/benchmark-program/benchmark-program.iml" />
|
||||||
<module fileurl="file://$PROJECT_DIR$/codeCore/codeCore.iml" filepath="$PROJECT_DIR$/codeCore/codeCore.iml" />
|
<module fileurl="file://$PROJECT_DIR$/codeCore/codeCore.iml" filepath="$PROJECT_DIR$/codeCore/codeCore.iml" />
|
||||||
<module fileurl="file://$PROJECT_DIR$/codeGenCpu6502/codeGenCpu6502.iml" filepath="$PROJECT_DIR$/codeGenCpu6502/codeGenCpu6502.iml" />
|
<module fileurl="file://$PROJECT_DIR$/codeGenCpu6502/codeGenCpu6502.iml" filepath="$PROJECT_DIR$/codeGenCpu6502/codeGenCpu6502.iml" />
|
||||||
<module fileurl="file://$PROJECT_DIR$/codeGenExperimental/codeGenExperimental.iml" filepath="$PROJECT_DIR$/codeGenExperimental/codeGenExperimental.iml" />
|
<module fileurl="file://$PROJECT_DIR$/codeGenExperimental/codeGenExperimental.iml" filepath="$PROJECT_DIR$/codeGenExperimental/codeGenExperimental.iml" />
|
||||||
@ -14,6 +16,7 @@
|
|||||||
<module fileurl="file://$PROJECT_DIR$/examples/examples.iml" filepath="$PROJECT_DIR$/examples/examples.iml" />
|
<module fileurl="file://$PROJECT_DIR$/examples/examples.iml" filepath="$PROJECT_DIR$/examples/examples.iml" />
|
||||||
<module fileurl="file://$PROJECT_DIR$/intermediate/intermediate.iml" filepath="$PROJECT_DIR$/intermediate/intermediate.iml" />
|
<module fileurl="file://$PROJECT_DIR$/intermediate/intermediate.iml" filepath="$PROJECT_DIR$/intermediate/intermediate.iml" />
|
||||||
<module fileurl="file://$PROJECT_DIR$/parser/parser.iml" filepath="$PROJECT_DIR$/parser/parser.iml" />
|
<module fileurl="file://$PROJECT_DIR$/parser/parser.iml" filepath="$PROJECT_DIR$/parser/parser.iml" />
|
||||||
|
<module fileurl="file://$PROJECT_DIR$/.idea/modules/prog8.iml" filepath="$PROJECT_DIR$/.idea/modules/prog8.iml" />
|
||||||
<module fileurl="file://$PROJECT_DIR$/virtualmachine/virtualmachine.iml" filepath="$PROJECT_DIR$/virtualmachine/virtualmachine.iml" />
|
<module fileurl="file://$PROJECT_DIR$/virtualmachine/virtualmachine.iml" filepath="$PROJECT_DIR$/virtualmachine/virtualmachine.iml" />
|
||||||
</modules>
|
</modules>
|
||||||
</component>
|
</component>
|
||||||
|
4
Makefile
4
Makefile
@ -4,9 +4,9 @@
|
|||||||
|
|
||||||
all:
|
all:
|
||||||
gradle installdist installshadowdist
|
gradle installdist installshadowdist
|
||||||
@echo "compiler launch script can be found here: compiler/build/install/compiler-shadow/bin/p8compile"
|
@echo "compiler launch script can be found here: compiler/build/install/compiler-shadow/bin/prog8c"
|
||||||
|
|
||||||
test:
|
test:
|
||||||
gradle build
|
gradle build
|
||||||
@echo "compiler launch script can be found here: compiler/build/install/compiler-shadow/bin/p8compile"
|
@echo "compiler launch script can be found here: compiler/build/install/compiler-shadow/bin/prog8c"
|
||||||
|
|
||||||
|
40
README.md
40
README.md
@ -29,6 +29,8 @@ How to get it/build it
|
|||||||
- Or, if you want/need a bleeding edge development version, you can:
|
- Or, if you want/need a bleeding edge development version, you can:
|
||||||
- download a build artifact zipfile from a recent [github action build](https://github.com/irmen/prog8/actions).
|
- download a build artifact zipfile from a recent [github action build](https://github.com/irmen/prog8/actions).
|
||||||
- you can also compile it yourself from source. [Instructions here](https://prog8.readthedocs.io/en/latest/compiling.html).
|
- you can also compile it yourself from source. [Instructions here](https://prog8.readthedocs.io/en/latest/compiling.html).
|
||||||
|
Note that if you are not using *gradle* to build it, you might have to perform some manual
|
||||||
|
tasks once to make it compile fully. These are explained in the linked instructions.
|
||||||
- Alternatively, you can also install the compiler as a package on some linux distros:
|
- Alternatively, you can also install the compiler as a package on some linux distros:
|
||||||
- Arch (via AUR): [`prog8`](https://aur.archlinux.org/packages/prog8)
|
- Arch (via AUR): [`prog8`](https://aur.archlinux.org/packages/prog8)
|
||||||
|
|
||||||
@ -52,26 +54,27 @@ What does Prog8 provide?
|
|||||||
------------------------
|
------------------------
|
||||||
|
|
||||||
- all advantages of a higher level language over having to write assembly code manually
|
- all advantages of a higher level language over having to write assembly code manually
|
||||||
- programs run very fast because compilation to native machine code. It's possible to write games purely in Prog8, and even certain raster interrupt 'demoscene' effects.
|
- programs run very fast because it's compiled to native machine code
|
||||||
- modularity, symbol scoping, subroutines
|
- code often is smaller and faster than equivalent C code compiled with CC65 or even LLVM-MOS
|
||||||
|
- modularity, symbol scoping, subroutines. No need for forward declarations.
|
||||||
- various data types other than just bytes (16-bit words, floats, strings)
|
- various data types other than just bytes (16-bit words, floats, strings)
|
||||||
- floating point math is supported if the target system provides floating point library routines (C64 and Cx16 both do)
|
- floating point math is supported on certain targets
|
||||||
|
- access to most Kernal ROM routines as external subroutine definitions you can call normally
|
||||||
- strings can contain escaped characters but also many symbols directly if they have a petscii equivalent, such as "♠♥♣♦π▚●○╳". Characters like ^, _, \, {, } and | are also accepted and converted to the closest petscii equivalents.
|
- strings can contain escaped characters but also many symbols directly if they have a petscii equivalent, such as "♠♥♣♦π▚●○╳". Characters like ^, _, \, {, } and | are also accepted and converted to the closest petscii equivalents.
|
||||||
- automatic static variable allocations, automatic string and array variables and string sharing
|
- automatic static variable allocations, automatic string and array variables and string sharing
|
||||||
- subroutines with input parameters and result values
|
|
||||||
- high-level program optimizations
|
- high-level program optimizations
|
||||||
- no need for forward declarations
|
|
||||||
- small program boilerplate/compilersupport overhead
|
|
||||||
- programs can be run multiple times without reloading because of automatic variable (re)initializations.
|
- programs can be run multiple times without reloading because of automatic variable (re)initializations.
|
||||||
- conditional branches
|
- conditional branches that map 1:1 to cpu status flags
|
||||||
- ``when`` statement to provide a concise jump table alternative to if/elseif chains
|
- ``when`` statement to provide a concise jump table alternative to if/elseif chains
|
||||||
- ``in`` expression for concise and efficient multi-value/containment check
|
- ``in`` expression for concise and efficient multi-value/containment check
|
||||||
- many built-in functions such as ``sin``, ``cos``, ``rnd``, ``abs``, ``min``, ``max``, ``sqrt``, ``msb``, ``rol``, ``ror``, ``swap``, ``sort`` and ``reverse``
|
- ``defer`` statement to help write concise and robust subroutine cleanup logic
|
||||||
|
- several specialized built-in functions such as ``lsb``, ``msb``, ``min``, ``max``, ``rol``, ``ror``
|
||||||
- various powerful built-in libraries to do I/O, number conversions, graphics and more
|
- various powerful built-in libraries to do I/O, number conversions, graphics and more
|
||||||
- convenience abstractions for low level aspects such as ZeroPage handling, program startup, explicit memory addresses
|
|
||||||
- inline assembly allows you to have full control when every cycle or byte matters
|
- inline assembly allows you to have full control when every cycle or byte matters
|
||||||
- supports the sixteen 'virtual' 16-bit registers R0 - R15 from the Commander X16, and provides them also on the C64.
|
- supports the sixteen 'virtual' 16-bit registers R0 - R15 from the Commander X16 (also available on other targets)
|
||||||
- encode strings and characters into petscii or screencodes as desired (C64/Cx16)
|
- encode strings and characters into petscii or screencodes or even other encodings
|
||||||
|
- Automatic ROM/RAM bank switching on certain compiler targets when calling routines in other banks
|
||||||
|
- 50 Kb of available program RAM size on the C64 by default; because Basic ROM is banked out altogether
|
||||||
|
|
||||||
*Rapid edit-compile-run-debug cycle:*
|
*Rapid edit-compile-run-debug cycle:*
|
||||||
|
|
||||||
@ -82,11 +85,12 @@ What does Prog8 provide?
|
|||||||
|
|
||||||
*Multiple supported compiler targets* (contributions to improve these or to add support for other machines are welcome!):
|
*Multiple supported compiler targets* (contributions to improve these or to add support for other machines are welcome!):
|
||||||
|
|
||||||
|
- "cx16": [CommanderX16](https://www.commanderx16.com) (65c02 CPU)
|
||||||
- "c64": Commodore-64 (6502 like CPU)
|
- "c64": Commodore-64 (6502 like CPU)
|
||||||
- "c128": Commodore-128 (6502 like CPU - the Z80 cpu mode is not supported)
|
- "c128": Commodore-128 (6502 like CPU - the Z80 cpu mode is not supported)
|
||||||
- "cx16": [CommanderX16](https://www.commanderx16.com) (65c02 CPU)
|
- "pet32": Commodore PET (limited support)
|
||||||
- "pet32": Commodore PET (experimental)
|
|
||||||
- "atari": Atari 8 bit such as 800XL (experimental)
|
- "atari": Atari 8 bit such as 800XL (experimental)
|
||||||
|
- "neo": Neo6502 (experimental)
|
||||||
- If you only use standard kernal and prog8 library routines, it is possible to compile the *exact same program* for different machines (just change the compiler target flag)
|
- If you only use standard kernal and prog8 library routines, it is possible to compile the *exact same program* for different machines (just change the compiler target flag)
|
||||||
|
|
||||||
|
|
||||||
@ -119,14 +123,13 @@ This code calculates prime numbers using the Sieve of Eratosthenes algorithm::
|
|||||||
|
|
||||||
%import textio
|
%import textio
|
||||||
%zeropage basicsafe
|
%zeropage basicsafe
|
||||||
|
|
||||||
main {
|
main {
|
||||||
|
bool[256] sieve
|
||||||
ubyte[256] sieve
|
|
||||||
ubyte candidate_prime = 2 ; is increased in the loop
|
ubyte candidate_prime = 2 ; is increased in the loop
|
||||||
|
|
||||||
sub start() {
|
sub start() {
|
||||||
sys.memset(sieve, 256, false) ; clear the sieve
|
sys.memset(sieve, 256, 0) ; clear the sieve
|
||||||
txt.print("prime numbers up to 255:\n\n")
|
txt.print("prime numbers up to 255:\n\n")
|
||||||
ubyte amount=0
|
ubyte amount=0
|
||||||
repeat {
|
repeat {
|
||||||
@ -162,9 +165,6 @@ This code calculates prime numbers using the Sieve of Eratosthenes algorithm::
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
when compiled an ran on a C-64 you'll get:
|
when compiled an ran on a C-64 you'll get:
|
||||||
|
|
||||||

|

|
||||||
|
24
beanshell/beanshell.iml
Normal file
24
beanshell/beanshell.iml
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<module type="JAVA_MODULE" version="4">
|
||||||
|
<component name="NewModuleRootManager" inherit-compiler-output="true">
|
||||||
|
<exclude-output />
|
||||||
|
<content url="file://$MODULE_DIR$">
|
||||||
|
<sourceFolder url="file://$MODULE_DIR$/src" isTestSource="false" />
|
||||||
|
<excludeFolder url="file://$MODULE_DIR$/build" />
|
||||||
|
</content>
|
||||||
|
<orderEntry type="inheritedJdk" />
|
||||||
|
<orderEntry type="sourceFolder" forTests="false" />
|
||||||
|
<orderEntry type="library" name="KotlinJavaRuntime" level="project" />
|
||||||
|
<orderEntry type="library" name="jetbrains.kotlinx.cli.jvm" level="project" />
|
||||||
|
<orderEntry type="library" name="michael.bull.kotlin.result.jvm" level="project" />
|
||||||
|
<orderEntry type="module-library">
|
||||||
|
<library>
|
||||||
|
<CLASSES>
|
||||||
|
<root url="jar://$MODULE_DIR$/lib/bsh-3.0.0-SNAPSHOT.jar!/" />
|
||||||
|
</CLASSES>
|
||||||
|
<JAVADOC />
|
||||||
|
<SOURCES />
|
||||||
|
</library>
|
||||||
|
</orderEntry>
|
||||||
|
</component>
|
||||||
|
</module>
|
65
beanshell/build.gradle.kts
Normal file
65
beanshell/build.gradle.kts
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
import org.jetbrains.kotlin.gradle.dsl.JvmTarget
|
||||||
|
|
||||||
|
plugins {
|
||||||
|
id("application")
|
||||||
|
kotlin("jvm")
|
||||||
|
}
|
||||||
|
|
||||||
|
val serverMainClassName = "prog8lsp.MainKt"
|
||||||
|
val applicationName = "prog8-beanshell"
|
||||||
|
|
||||||
|
application {
|
||||||
|
mainClass.set(serverMainClassName)
|
||||||
|
description = "Code completions, diagnostics and more for Prog8"
|
||||||
|
// applicationDefaultJvmArgs = listOf("-DkotlinLanguageServer.version=$version")
|
||||||
|
applicationDistribution.into("bin") {
|
||||||
|
filePermissions {
|
||||||
|
user {
|
||||||
|
read=true
|
||||||
|
execute=true
|
||||||
|
write=true
|
||||||
|
}
|
||||||
|
other.execute = true
|
||||||
|
group.execute = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
mavenCentral()
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
implementation(files("lib/bsh-3.0.0-SNAPSHOT.jar"))
|
||||||
|
}
|
||||||
|
|
||||||
|
configurations.forEach { config ->
|
||||||
|
config.resolutionStrategy {
|
||||||
|
preferProjectModules()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sourceSets.main {
|
||||||
|
java.srcDir("src")
|
||||||
|
resources.srcDir("resources")
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.startScripts {
|
||||||
|
applicationName = "prog8-beanshell"
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.register<Exec>("fixFilePermissions") {
|
||||||
|
// When running on macOS or Linux the start script
|
||||||
|
// needs executable permissions to run.
|
||||||
|
|
||||||
|
onlyIf { !System.getProperty("os.name").lowercase().contains("windows") }
|
||||||
|
commandLine("chmod", "+x", "${tasks.installDist.get().destinationDir}/bin/prog8-beanshell")
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.installDist {
|
||||||
|
finalizedBy("fixFilePermissions")
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.build {
|
||||||
|
finalizedBy("installDist")
|
||||||
|
}
|
BIN
beanshell/lib/bsh-3.0.0-SNAPSHOT.jar
Normal file
BIN
beanshell/lib/bsh-3.0.0-SNAPSHOT.jar
Normal file
Binary file not shown.
48
beanshell/src/prog8beanshell/CommandLineReader.kt
Normal file
48
beanshell/src/prog8beanshell/CommandLineReader.kt
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
package prog8beanshell
|
||||||
|
|
||||||
|
import java.io.FilterReader
|
||||||
|
import java.io.Reader
|
||||||
|
|
||||||
|
|
||||||
|
class CommandLineReader(val input: Reader): FilterReader(input) {
|
||||||
|
private val normal = 0
|
||||||
|
private val lastCharNL = 1
|
||||||
|
private val sentSemi = 2
|
||||||
|
private var state = lastCharNL
|
||||||
|
|
||||||
|
override fun read(): Int {
|
||||||
|
if (state == sentSemi) {
|
||||||
|
this.state = lastCharNL
|
||||||
|
return 10
|
||||||
|
} else {
|
||||||
|
var b = input.read()
|
||||||
|
while(b==13) b = input.read()
|
||||||
|
|
||||||
|
if (b == 10) {
|
||||||
|
if (this.state == lastCharNL) {
|
||||||
|
b = 59
|
||||||
|
this.state = sentSemi
|
||||||
|
} else {
|
||||||
|
this.state = lastCharNL
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.state = normal
|
||||||
|
}
|
||||||
|
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun read(buff: CharArray, off: Int, len: Int): Int {
|
||||||
|
val b = read()
|
||||||
|
if (b == -1) {
|
||||||
|
return -1
|
||||||
|
} else {
|
||||||
|
buff[off] = b.toChar()
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
23
beanshell/src/prog8beanshell/Main.kt
Normal file
23
beanshell/src/prog8beanshell/Main.kt
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
package prog8beanshell
|
||||||
|
|
||||||
|
import bsh.FileReader
|
||||||
|
import bsh.Interpreter
|
||||||
|
|
||||||
|
|
||||||
|
class BeanshellInterpreter {
|
||||||
|
|
||||||
|
fun run(symbols: Map<String, Any>) {
|
||||||
|
val interpreter = Interpreter(CommandLineReader(FileReader(System.`in`)), System.out, System.err, true)
|
||||||
|
interpreter.setExitOnEOF(false)
|
||||||
|
symbols.forEach { (name, value) -> interpreter.set(name, value) }
|
||||||
|
interpreter.run()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun main(args: Array<String>) {
|
||||||
|
val i = BeanshellInterpreter()
|
||||||
|
i.run(mapOf(
|
||||||
|
"env" to System.getenv(),
|
||||||
|
"args" to args
|
||||||
|
))
|
||||||
|
}
|
10
benchmark-program/Makefile
Normal file
10
benchmark-program/Makefile
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
.PHONY: clean run
|
||||||
|
|
||||||
|
run:
|
||||||
|
prog8c -target cx16 benchmark.p8
|
||||||
|
x16emu -run -prg benchmark.prg -warp
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -f *.prg *.PRG *.asm *.vice-* *.BIN *.PAL *.zip *.7z
|
||||||
|
|
||||||
|
|
109
benchmark-program/b_3d.p8
Normal file
109
benchmark-program/b_3d.p8
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
%import textio
|
||||||
|
%import math
|
||||||
|
|
||||||
|
rotate3d {
|
||||||
|
const ubyte WIDTH = 40
|
||||||
|
const ubyte HEIGHT = 30
|
||||||
|
|
||||||
|
sub benchmark(uword max_time) -> uword {
|
||||||
|
|
||||||
|
uword anglex
|
||||||
|
uword angley
|
||||||
|
uword anglez
|
||||||
|
uword frames
|
||||||
|
|
||||||
|
txt.nl()
|
||||||
|
cbm.SETTIM(0,0,0)
|
||||||
|
|
||||||
|
while cbm.RDTIM16()<max_time {
|
||||||
|
matrix_math.rotate_vertices(msb(anglex), msb(angley), msb(anglez))
|
||||||
|
draw_edges() ; doesn't really draw anything in the benchmark, but does do the screen calculations
|
||||||
|
anglex+=500
|
||||||
|
angley+=215
|
||||||
|
anglez+=453
|
||||||
|
frames++
|
||||||
|
}
|
||||||
|
|
||||||
|
return frames
|
||||||
|
}
|
||||||
|
|
||||||
|
sub draw_edges() {
|
||||||
|
|
||||||
|
; plot the points of the 3d cube
|
||||||
|
; first the points on the back, then the points on the front (painter algorithm)
|
||||||
|
|
||||||
|
ubyte @zp i
|
||||||
|
word @zp rz
|
||||||
|
word @zp persp
|
||||||
|
byte @shared sx
|
||||||
|
byte @shared sy
|
||||||
|
|
||||||
|
for i in 0 to len(matrix_math.xcoor)-1 {
|
||||||
|
rz = matrix_math.rotatedz[i]
|
||||||
|
if rz >= 10 {
|
||||||
|
persp = 600 + rz/64
|
||||||
|
sx = matrix_math.rotatedx[i] / persp as byte + WIDTH/2
|
||||||
|
sy = matrix_math.rotatedy[i] / persp as byte + HEIGHT/2
|
||||||
|
;; txt.setcc(sx as ubyte, sy as ubyte, 46, 7)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for i in 0 to len(matrix_math.xcoor)-1 {
|
||||||
|
rz = matrix_math.rotatedz[i]
|
||||||
|
if rz < 10 {
|
||||||
|
persp = 600 + rz/64
|
||||||
|
sx = matrix_math.rotatedx[i] / persp as byte + WIDTH/2
|
||||||
|
sy = matrix_math.rotatedy[i] / persp as byte + HEIGHT/2
|
||||||
|
;; txt.setcc(sx as ubyte, sy as ubyte, 81, 7)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
txt.chrout('.')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
matrix_math {
|
||||||
|
; vertices
|
||||||
|
word[] xcoor = [ -40, -40, -40, -40, 40, 40, 40, 40 ]
|
||||||
|
word[] ycoor = [ -40, -40, 40, 40, -40, -40, 40, 40 ]
|
||||||
|
word[] zcoor = [ -40, 40, -40, 40, -40, 40, -40, 40 ]
|
||||||
|
|
||||||
|
; storage for rotated coordinates
|
||||||
|
word[len(xcoor)] rotatedx
|
||||||
|
word[len(ycoor)] rotatedy
|
||||||
|
word[len(zcoor)] rotatedz
|
||||||
|
|
||||||
|
sub rotate_vertices(ubyte ax, ubyte ay, ubyte az) {
|
||||||
|
; rotate around origin (0,0,0)
|
||||||
|
|
||||||
|
; set up the 3d rotation matrix values
|
||||||
|
word wcosa = math.cos8(ax)
|
||||||
|
word wsina = math.sin8(ax)
|
||||||
|
word wcosb = math.cos8(ay)
|
||||||
|
word wsinb = math.sin8(ay)
|
||||||
|
word wcosc = math.cos8(az)
|
||||||
|
word wsinc = math.sin8(az)
|
||||||
|
|
||||||
|
word wcosa_sinb = wcosa*wsinb / 128
|
||||||
|
word wsina_sinb = wsina*wsinb / 128
|
||||||
|
|
||||||
|
word Axx = wcosa*wcosb / 128
|
||||||
|
word Axy = (wcosa_sinb*wsinc - wsina*wcosc) / 128
|
||||||
|
word Axz = (wcosa_sinb*wcosc + wsina*wsinc) / 128
|
||||||
|
word Ayx = wsina*wcosb / 128
|
||||||
|
word Ayy = (wsina_sinb*wsinc + wcosa*wcosc) / 128
|
||||||
|
word Ayz = (wsina_sinb*wcosc - wcosa*wsinc) / 128
|
||||||
|
word Azx = -wsinb
|
||||||
|
word Azy = wcosb*wsinc / 128
|
||||||
|
word Azz = wcosb*wcosc / 128
|
||||||
|
|
||||||
|
ubyte @zp i
|
||||||
|
for i in 0 to len(xcoor)-1 {
|
||||||
|
; don't normalize by dividing by 128, instead keep some precision for perspective calc later
|
||||||
|
rotatedx[i] = Axx*xcoor[i] + Axy*ycoor[i] + Axz*zcoor[i]
|
||||||
|
rotatedy[i] = Ayx*xcoor[i] + Ayy*ycoor[i] + Ayz*zcoor[i]
|
||||||
|
rotatedz[i] = Azx*xcoor[i] + Azy*ycoor[i] + Azz*zcoor[i]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
89
benchmark-program/b_adpcm.p8
Normal file
89
benchmark-program/b_adpcm.p8
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
adpcm {
|
||||||
|
|
||||||
|
sub decode_benchmark(uword max_time) -> uword {
|
||||||
|
uword num_blocks
|
||||||
|
txt.nl()
|
||||||
|
cbm.SETTIM(0,0,0)
|
||||||
|
|
||||||
|
while cbm.RDTIM16()<max_time {
|
||||||
|
adpcm.init(0,0)
|
||||||
|
uword @requirezp nibbles_ptr = $a000 ; for benchmark purposes, the exact nibbles don't really matter, so we just take the basic ROM as input
|
||||||
|
repeat 252/2 {
|
||||||
|
unroll 2 {
|
||||||
|
ubyte @zp nibble = @(nibbles_ptr)
|
||||||
|
adpcm.decode_nibble(nibble & 15) ; first word (note: upper nibble needs to be zero!)
|
||||||
|
adpcm.decode_nibble(nibble>>4) ; second word (note: upper nibble is zero, after the shifts.)
|
||||||
|
nibbles_ptr++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
num_blocks++
|
||||||
|
txt.chrout('.')
|
||||||
|
}
|
||||||
|
|
||||||
|
return num_blocks
|
||||||
|
}
|
||||||
|
|
||||||
|
; IMA ADPCM decoder. Supports mono and stereo streams.
|
||||||
|
|
||||||
|
byte[] t_index = [ -1, -1, -1, -1, 2, 4, 6, 8, -1, -1, -1, -1, 2, 4, 6, 8]
|
||||||
|
uword[] t_step = [
|
||||||
|
7, 8, 9, 10, 11, 12, 13, 14,
|
||||||
|
16, 17, 19, 21, 23, 25, 28, 31,
|
||||||
|
34, 37, 41, 45, 50, 55, 60, 66,
|
||||||
|
73, 80, 88, 97, 107, 118, 130, 143,
|
||||||
|
157, 173, 190, 209, 230, 253, 279, 307,
|
||||||
|
337, 371, 408, 449, 494, 544, 598, 658,
|
||||||
|
724, 796, 876, 963, 1060, 1166, 1282, 1411,
|
||||||
|
1552, 1707, 1878, 2066, 2272, 2499, 2749, 3024,
|
||||||
|
3327, 3660, 4026, 4428, 4871, 5358, 5894, 6484,
|
||||||
|
7132, 7845, 8630, 9493, 10442, 11487, 12635, 13899,
|
||||||
|
15289, 16818, 18500, 20350, 22385, 24623, 27086, 29794,
|
||||||
|
32767]
|
||||||
|
|
||||||
|
uword @requirezp predict ; decoded 16 bit pcm sample for first channel.
|
||||||
|
ubyte @requirezp index
|
||||||
|
uword @requirezp pstep
|
||||||
|
|
||||||
|
sub init(uword startPredict, ubyte startIndex) {
|
||||||
|
; initialize first decoding channel.
|
||||||
|
predict = startPredict
|
||||||
|
index = startIndex
|
||||||
|
pstep = t_step[index]
|
||||||
|
}
|
||||||
|
|
||||||
|
sub decode_nibble(ubyte @zp nibble) {
|
||||||
|
; Decoder for a single nibble for the first channel. (value of 'nibble' needs to be strictly 0-15 !)
|
||||||
|
; This is the hotspot of the decoder algorithm!
|
||||||
|
; Note that the generated assembly from this is pretty efficient,
|
||||||
|
; rewriting it by hand in asm seems to improve it only ~10%.
|
||||||
|
cx16.r0s = 0 ; difference
|
||||||
|
if nibble & %0100 !=0
|
||||||
|
cx16.r0s += pstep
|
||||||
|
pstep >>= 1
|
||||||
|
if nibble & %0010 !=0
|
||||||
|
cx16.r0s += pstep
|
||||||
|
pstep >>= 1
|
||||||
|
if nibble & %0001 !=0
|
||||||
|
cx16.r0s += pstep
|
||||||
|
pstep >>= 1
|
||||||
|
cx16.r0s += pstep
|
||||||
|
if nibble & %1000 !=0
|
||||||
|
predict -= cx16.r0
|
||||||
|
else
|
||||||
|
predict += cx16.r0
|
||||||
|
|
||||||
|
; NOTE: the original C/Python code uses a 32 bits prediction value and clips it to a 16 bit word
|
||||||
|
; but for speed reasons we only work with 16 bit words here all the time (with possible clipping error)
|
||||||
|
; if predicted > 32767:
|
||||||
|
; predicted = 32767
|
||||||
|
; elif predicted < -32767:
|
||||||
|
; predicted = - 32767
|
||||||
|
|
||||||
|
index += t_index[nibble] as ubyte
|
||||||
|
if_neg
|
||||||
|
index = 0
|
||||||
|
else if index >= len(t_step)-1
|
||||||
|
index = len(t_step)-1
|
||||||
|
pstep = t_step[index]
|
||||||
|
}
|
||||||
|
}
|
111
benchmark-program/b_circles.p8
Normal file
111
benchmark-program/b_circles.p8
Normal file
@ -0,0 +1,111 @@
|
|||||||
|
%import gfx_lores
|
||||||
|
%import math
|
||||||
|
|
||||||
|
circles {
|
||||||
|
const ubyte MAX_NUM_CIRCLES = 80
|
||||||
|
const ubyte GROWTH_RATE = 4
|
||||||
|
uword[MAX_NUM_CIRCLES] circle_x
|
||||||
|
uword[MAX_NUM_CIRCLES] circle_y
|
||||||
|
ubyte[MAX_NUM_CIRCLES] circle_radius
|
||||||
|
ubyte color
|
||||||
|
uword total_num_circles
|
||||||
|
|
||||||
|
sub draw(bool use_kernal, uword max_time) -> uword {
|
||||||
|
if use_kernal
|
||||||
|
void cx16.set_screen_mode(128)
|
||||||
|
else
|
||||||
|
gfx_lores.graphics_mode()
|
||||||
|
|
||||||
|
math.rndseed(12345,6789)
|
||||||
|
cbm.SETTIM(0,0,0)
|
||||||
|
|
||||||
|
total_num_circles = 0
|
||||||
|
color = 16
|
||||||
|
|
||||||
|
while cbm.RDTIM16()<max_time {
|
||||||
|
if use_kernal {
|
||||||
|
cx16.GRAPH_set_colors(0,0,0)
|
||||||
|
cx16.GRAPH_clear()
|
||||||
|
}
|
||||||
|
else
|
||||||
|
gfx_lores.clear_screen(0)
|
||||||
|
total_num_circles += draw_circles(use_kernal, max_time)
|
||||||
|
}
|
||||||
|
|
||||||
|
if use_kernal
|
||||||
|
void cx16.set_screen_mode(3)
|
||||||
|
else {
|
||||||
|
gfx_lores.text_mode()
|
||||||
|
}
|
||||||
|
|
||||||
|
return total_num_circles
|
||||||
|
}
|
||||||
|
|
||||||
|
sub draw_circles(bool use_kernal, uword max_time) -> uword {
|
||||||
|
uword @zp x
|
||||||
|
uword @zp y
|
||||||
|
ubyte @zp radius
|
||||||
|
|
||||||
|
ubyte num_circles
|
||||||
|
|
||||||
|
while num_circles<MAX_NUM_CIRCLES and cbm.RDTIM16()<max_time {
|
||||||
|
x = math.rndw() % 320
|
||||||
|
y = math.rndw() % 240
|
||||||
|
radius = GROWTH_RATE
|
||||||
|
if not_colliding() {
|
||||||
|
while not_edge() and not_colliding() {
|
||||||
|
radius += GROWTH_RATE
|
||||||
|
}
|
||||||
|
radius -= GROWTH_RATE
|
||||||
|
if radius>0 {
|
||||||
|
color++
|
||||||
|
if color==0
|
||||||
|
color=16
|
||||||
|
if use_kernal {
|
||||||
|
cx16.GRAPH_set_colors(color, 255-color, 0)
|
||||||
|
cx16.GRAPH_draw_oval(x-radius, y-radius, radius*2, radius*2, true)
|
||||||
|
}
|
||||||
|
else
|
||||||
|
gfx_lores.disc(x, y as ubyte, radius, color)
|
||||||
|
circle_x[num_circles] = x
|
||||||
|
circle_y[num_circles] = y
|
||||||
|
circle_radius[num_circles] = radius
|
||||||
|
num_circles++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return num_circles
|
||||||
|
|
||||||
|
sub not_colliding() -> bool {
|
||||||
|
if num_circles==0
|
||||||
|
return true
|
||||||
|
ubyte @zp c
|
||||||
|
for c in 0 to num_circles-1 {
|
||||||
|
if distance(c) < (radius as uword) + circle_radius[c]
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
sub distance(ubyte cix) -> uword {
|
||||||
|
word dx = x as word - circle_x[cix]
|
||||||
|
word dy = y as word - circle_y[cix]
|
||||||
|
uword sqx = dx*dx as uword
|
||||||
|
uword sqy = dy*dy as uword
|
||||||
|
return sqrt(sqx + sqy)
|
||||||
|
}
|
||||||
|
|
||||||
|
sub not_edge() -> bool {
|
||||||
|
if x as word - radius < 0
|
||||||
|
return false
|
||||||
|
if x + radius >= 320
|
||||||
|
return false
|
||||||
|
if y as word - radius < 0
|
||||||
|
return false
|
||||||
|
if y + radius >= 240
|
||||||
|
return false
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
123
benchmark-program/b_life.p8
Normal file
123
benchmark-program/b_life.p8
Normal file
@ -0,0 +1,123 @@
|
|||||||
|
; conway's game of life.
|
||||||
|
|
||||||
|
%import math
|
||||||
|
%import textio
|
||||||
|
|
||||||
|
life {
|
||||||
|
const ubyte WIDTH = 40
|
||||||
|
const ubyte HEIGHT = 30
|
||||||
|
const uword STRIDE = $0002+WIDTH
|
||||||
|
uword world1 = memory("world1", (WIDTH+2)*(HEIGHT+2), 0)
|
||||||
|
uword world2 = memory("world2", (WIDTH+2)*(HEIGHT+2), 0)
|
||||||
|
uword @requirezp active_world = world1
|
||||||
|
|
||||||
|
sub benchmark(uword max_time) -> uword {
|
||||||
|
txt.clear_screen()
|
||||||
|
sys.memset(world1, (WIDTH+2)*(HEIGHT+2), 0)
|
||||||
|
sys.memset(world2, (WIDTH+2)*(HEIGHT+2), 0)
|
||||||
|
|
||||||
|
set_start_gen()
|
||||||
|
|
||||||
|
uword gen
|
||||||
|
cbm.SETTIM(0,0,0)
|
||||||
|
|
||||||
|
while cbm.RDTIM16()<max_time {
|
||||||
|
next_gen()
|
||||||
|
gen++
|
||||||
|
}
|
||||||
|
|
||||||
|
return gen
|
||||||
|
}
|
||||||
|
|
||||||
|
sub set_start_gen() {
|
||||||
|
|
||||||
|
; some way to set a custom start generation:
|
||||||
|
; str start_gen = " " +
|
||||||
|
; " " +
|
||||||
|
; " " +
|
||||||
|
; " ** " +
|
||||||
|
; " * * " +
|
||||||
|
; " * " +
|
||||||
|
; " * * " +
|
||||||
|
; " ****** " +
|
||||||
|
; " " +
|
||||||
|
; " " +
|
||||||
|
; " " +
|
||||||
|
; " " +
|
||||||
|
; " " +
|
||||||
|
; " " +
|
||||||
|
; " " +
|
||||||
|
; " "
|
||||||
|
;
|
||||||
|
; for y in 0 to 15 {
|
||||||
|
; for x in 0 to 15 {
|
||||||
|
; if start_gen[y*16 + x]=='*'
|
||||||
|
; active_world[offset + x] = 1
|
||||||
|
; }
|
||||||
|
; offset += STRIDE
|
||||||
|
; }
|
||||||
|
|
||||||
|
; randomize whole world
|
||||||
|
math.rndseed(12345,9999)
|
||||||
|
uword offset = STRIDE+1
|
||||||
|
ubyte x
|
||||||
|
ubyte y
|
||||||
|
for y in 0 to HEIGHT-1 {
|
||||||
|
for x in 0 to WIDTH-1 {
|
||||||
|
active_world[offset+x] = math.rnd() & 1
|
||||||
|
}
|
||||||
|
offset += STRIDE
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub next_gen() {
|
||||||
|
const ubyte DXOFFSET = 0
|
||||||
|
const ubyte DYOFFSET = 0
|
||||||
|
ubyte[2] cell_chars = [sc:' ', sc:'●']
|
||||||
|
|
||||||
|
uword @requirezp new_world = world1
|
||||||
|
if active_world == world1
|
||||||
|
new_world = world2
|
||||||
|
|
||||||
|
; To avoid re-calculating word index lookups into the new- and active world arrays,
|
||||||
|
; we calculate the required pointer values upfront.
|
||||||
|
; Inside the loop we can use ptr+x just fine (results in efficient LDA (ptr),Y instruction because x is a byte type),
|
||||||
|
; and for each row we simply add the stride to the pointer.
|
||||||
|
; It's more readable to use active_world[offset] etc, but offset is a word value, and this produces
|
||||||
|
; inefficient assembly code because we can't use a register indexed mode in this case. Costly inside a loop.
|
||||||
|
|
||||||
|
uword @requirezp new_world_ptr = new_world + STRIDE+1-DXOFFSET
|
||||||
|
uword @requirezp active_world_ptr = active_world + STRIDE+1-DXOFFSET
|
||||||
|
|
||||||
|
ubyte x
|
||||||
|
ubyte y
|
||||||
|
for y in DYOFFSET to HEIGHT+DYOFFSET-1 {
|
||||||
|
|
||||||
|
cx16.vaddr_autoincr(1, $b000 + 256*y, 0, 2) ; allows us to use simple Vera data byte assigns later instead of setchr() calls
|
||||||
|
|
||||||
|
for x in DXOFFSET to WIDTH+DXOFFSET-1 {
|
||||||
|
; count the living neighbors
|
||||||
|
ubyte cell = @(active_world_ptr + x)
|
||||||
|
uword @requirezp ptr = active_world_ptr + x - STRIDE - 1
|
||||||
|
ubyte neighbors = @(ptr) + @(ptr+1) + @(ptr+2) +
|
||||||
|
@(ptr+STRIDE) + cell + @(ptr+STRIDE+2) +
|
||||||
|
@(ptr+STRIDE*2) + @(ptr+STRIDE*2+1) + @(ptr+STRIDE*2+2)
|
||||||
|
|
||||||
|
; apply game of life rules
|
||||||
|
if neighbors==3
|
||||||
|
cell=1
|
||||||
|
else if neighbors!=4
|
||||||
|
cell=0
|
||||||
|
@(new_world_ptr + x) = cell
|
||||||
|
|
||||||
|
; draw new cell
|
||||||
|
; txt.setchr(x,y,cell_chars[cell])
|
||||||
|
cx16.VERA_DATA0 = cell_chars[cell]
|
||||||
|
}
|
||||||
|
active_world_ptr += STRIDE
|
||||||
|
new_world_ptr += STRIDE
|
||||||
|
}
|
||||||
|
|
||||||
|
active_world = new_world
|
||||||
|
}
|
||||||
|
}
|
54
benchmark-program/b_mandelbrot.p8
Normal file
54
benchmark-program/b_mandelbrot.p8
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
%import textio
|
||||||
|
%import floats
|
||||||
|
|
||||||
|
mandelbrot {
|
||||||
|
const ubyte width = 39
|
||||||
|
const ubyte height = 29
|
||||||
|
const ubyte max_iter = 15
|
||||||
|
|
||||||
|
sub calc(uword max_time) -> uword {
|
||||||
|
uword num_pixels
|
||||||
|
ubyte pixelx
|
||||||
|
ubyte pixely
|
||||||
|
|
||||||
|
txt.home()
|
||||||
|
cbm.SETTIM(0,0,0)
|
||||||
|
|
||||||
|
while cbm.RDTIM16() < max_time {
|
||||||
|
for pixely in 0 to height-1 {
|
||||||
|
float yy = (pixely as float)/0.40/height - 1.3
|
||||||
|
|
||||||
|
for pixelx in 0 to width-1 {
|
||||||
|
float xx = (pixelx as float)/0.32/width - 2.2
|
||||||
|
|
||||||
|
float xsquared = 0.0
|
||||||
|
float ysquared = 0.0
|
||||||
|
float x = 0.0
|
||||||
|
float y = 0.0
|
||||||
|
ubyte iter = 0
|
||||||
|
|
||||||
|
while iter<max_iter and xsquared+ysquared<4.0 {
|
||||||
|
y = x*y*2.0 + yy
|
||||||
|
x = xsquared - ysquared + xx
|
||||||
|
xsquared = x*x
|
||||||
|
ysquared = y*y
|
||||||
|
iter++
|
||||||
|
}
|
||||||
|
txt.color2(1, max_iter-iter)
|
||||||
|
txt.spc()
|
||||||
|
num_pixels++
|
||||||
|
|
||||||
|
if cbm.RDTIM16()>=max_time
|
||||||
|
goto finished
|
||||||
|
}
|
||||||
|
txt.nl()
|
||||||
|
}
|
||||||
|
|
||||||
|
txt.clear_screen()
|
||||||
|
}
|
||||||
|
|
||||||
|
finished:
|
||||||
|
txt.color2(1, 6)
|
||||||
|
return num_pixels
|
||||||
|
}
|
||||||
|
}
|
343
benchmark-program/b_maze.p8
Normal file
343
benchmark-program/b_maze.p8
Normal file
@ -0,0 +1,343 @@
|
|||||||
|
%import textio
|
||||||
|
%import math
|
||||||
|
|
||||||
|
; Even though prog8 only has support for extremely limited recursion,
|
||||||
|
; you can write recursive algorithms with a bit of extra work by building your own explicit stack structure.
|
||||||
|
; This program shows a depth-first maze generation algorithm (1 possible path from start to finish),
|
||||||
|
; and a depth-first maze solver algorithm, both using a stack to store the path taken.
|
||||||
|
|
||||||
|
; Note: this program can be compiled for multiple target systems.
|
||||||
|
|
||||||
|
maze {
|
||||||
|
uword score
|
||||||
|
|
||||||
|
sub bench(uword max_time) -> uword {
|
||||||
|
txt.nl()
|
||||||
|
score=0
|
||||||
|
math.rndseed(2345,44332)
|
||||||
|
cbm.SETTIM(0,0,0)
|
||||||
|
while cbm.RDTIM16()<max_time {
|
||||||
|
maze.initialize()
|
||||||
|
maze.drawStartFinish()
|
||||||
|
if maze.generate(max_time) {
|
||||||
|
maze.openpassages()
|
||||||
|
maze.drawStartFinish()
|
||||||
|
if maze.solve(max_time) {
|
||||||
|
maze.drawStartFinish()
|
||||||
|
} else break
|
||||||
|
} else break
|
||||||
|
}
|
||||||
|
|
||||||
|
return score
|
||||||
|
}
|
||||||
|
|
||||||
|
const uword screenwidth = 40
|
||||||
|
const uword screenheight = 30
|
||||||
|
|
||||||
|
const ubyte numCellsHoriz = (screenwidth-1) / 2
|
||||||
|
const ubyte numCellsVert = (screenheight-1) / 2
|
||||||
|
|
||||||
|
; maze start and finish cells
|
||||||
|
const ubyte startCx = 0
|
||||||
|
const ubyte startCy = 0
|
||||||
|
const ubyte finishCx = numCellsHoriz-1
|
||||||
|
const ubyte finishCy = numCellsVert-1
|
||||||
|
|
||||||
|
; cell properties
|
||||||
|
const ubyte STONE = 128
|
||||||
|
const ubyte WALKED = 64
|
||||||
|
const ubyte BACKTRACKED = 32
|
||||||
|
const ubyte UP = 1
|
||||||
|
const ubyte RIGHT = 2
|
||||||
|
const ubyte DOWN = 4
|
||||||
|
const ubyte LEFT = 8
|
||||||
|
const ubyte WALLCOLOR = 12
|
||||||
|
const ubyte EMPTYCOLOR = 0
|
||||||
|
|
||||||
|
; unfortunately on larger screens (cx16), the number of cells exceeds 256 and doesn't fit in a regular array anymore.
|
||||||
|
uword cells = memory("cells", numCellsHoriz*numCellsVert, 0)
|
||||||
|
|
||||||
|
ubyte[256] cx_stack
|
||||||
|
ubyte[256] cy_stack
|
||||||
|
ubyte stackptr
|
||||||
|
|
||||||
|
ubyte[4] directionflags = [LEFT,RIGHT,UP,DOWN]
|
||||||
|
|
||||||
|
sub generate(uword max_time) -> bool {
|
||||||
|
ubyte cx = startCx
|
||||||
|
ubyte cy = startCy
|
||||||
|
|
||||||
|
stackptr = 0
|
||||||
|
@(celladdr(cx,cy)) &= ~STONE
|
||||||
|
drawCell(cx, cy)
|
||||||
|
uword cells_to_carve = numCellsHoriz * numCellsVert - 1
|
||||||
|
|
||||||
|
while cbm.RDTIM16()<max_time {
|
||||||
|
carve_restart_after_repath:
|
||||||
|
ubyte direction = choose_uncarved_direction()
|
||||||
|
if direction==0 {
|
||||||
|
;backtrack
|
||||||
|
stackptr--
|
||||||
|
if stackptr==255 {
|
||||||
|
; stack empty.
|
||||||
|
; repath if we are not done yet. (this is a workaround for the prog8 256 array lenght limit)
|
||||||
|
if cells_to_carve!=0 {
|
||||||
|
if repath()
|
||||||
|
goto carve_restart_after_repath
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
cx = cx_stack[stackptr]
|
||||||
|
cy = cy_stack[stackptr]
|
||||||
|
} else {
|
||||||
|
cx_stack[stackptr] = cx
|
||||||
|
cy_stack[stackptr] = cy
|
||||||
|
stackptr++
|
||||||
|
if stackptr==0 {
|
||||||
|
; stack overflow, we can't track our path any longer.
|
||||||
|
; repath if we are not done yet. (this is a workaround for the prog8 256 array lenght limit)
|
||||||
|
if cells_to_carve!=0 {
|
||||||
|
if repath()
|
||||||
|
goto carve_restart_after_repath
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
@(celladdr(cx,cy)) |= direction
|
||||||
|
when direction {
|
||||||
|
UP -> {
|
||||||
|
cy--
|
||||||
|
@(celladdr(cx,cy)) |= DOWN
|
||||||
|
}
|
||||||
|
RIGHT -> {
|
||||||
|
cx++
|
||||||
|
@(celladdr(cx,cy)) |= LEFT
|
||||||
|
|
||||||
|
score++
|
||||||
|
}
|
||||||
|
DOWN -> {
|
||||||
|
cy++
|
||||||
|
@(celladdr(cx,cy)) |= UP
|
||||||
|
}
|
||||||
|
LEFT -> {
|
||||||
|
cx--
|
||||||
|
@(celladdr(cx,cy)) |= RIGHT
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@(celladdr(cx,cy)) &= ~STONE
|
||||||
|
cells_to_carve--
|
||||||
|
drawCell(cx, cy)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
|
||||||
|
sub repath() -> bool {
|
||||||
|
; repath: try to find a new start cell with possible directions.
|
||||||
|
; we limit our number of searches so that the algorith doesn't get stuck
|
||||||
|
; for too long on bad rng... just accept a few unused cells in that case.
|
||||||
|
repeat 255 {
|
||||||
|
do {
|
||||||
|
cx = math.rnd() % numCellsHoriz
|
||||||
|
cy = math.rnd() % numCellsVert
|
||||||
|
} until @(celladdr(cx, cy)) & STONE ==0
|
||||||
|
if available_uncarved()!=0
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
sub available_uncarved() -> ubyte {
|
||||||
|
ubyte candidates = 0
|
||||||
|
if cx>0 and @(celladdr(cx-1, cy)) & STONE !=0
|
||||||
|
candidates |= LEFT
|
||||||
|
if cx<numCellsHoriz-1 and @(celladdr(cx+1, cy)) & STONE !=0
|
||||||
|
candidates |= RIGHT
|
||||||
|
if cy>0 and @(celladdr(cx, cy-1)) & STONE !=0
|
||||||
|
candidates |= UP
|
||||||
|
if cy<numCellsVert-1 and @(celladdr(cx, cy+1)) & STONE !=0
|
||||||
|
candidates |= DOWN
|
||||||
|
return candidates
|
||||||
|
}
|
||||||
|
|
||||||
|
sub choose_uncarved_direction() -> ubyte {
|
||||||
|
ubyte candidates = available_uncarved()
|
||||||
|
if candidates==0
|
||||||
|
return 0
|
||||||
|
|
||||||
|
repeat {
|
||||||
|
ubyte choice = candidates & directionflags[math.rnd() & 3]
|
||||||
|
if choice!=0
|
||||||
|
return choice
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub openpassages() {
|
||||||
|
; open just a few extra passages, so that multiple routes are possible in theory.
|
||||||
|
ubyte numpassages
|
||||||
|
ubyte cx
|
||||||
|
ubyte cy
|
||||||
|
do {
|
||||||
|
do {
|
||||||
|
cx = math.rnd() % (numCellsHoriz-2) + 1
|
||||||
|
cy = math.rnd() % (numCellsVert-2) + 1
|
||||||
|
} until @(celladdr(cx, cy)) & STONE ==0
|
||||||
|
ubyte direction = directionflags[math.rnd() & 3]
|
||||||
|
if @(celladdr(cx, cy)) & direction == 0 {
|
||||||
|
when direction {
|
||||||
|
LEFT -> {
|
||||||
|
if @(celladdr(cx-1,cy)) & STONE == 0 {
|
||||||
|
@(celladdr(cx,cy)) |= LEFT
|
||||||
|
drawCell(cx,cy)
|
||||||
|
numpassages++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
RIGHT -> {
|
||||||
|
if @(celladdr(cx+1,cy)) & STONE == 0 {
|
||||||
|
@(celladdr(cx,cy)) |= RIGHT
|
||||||
|
drawCell(cx,cy)
|
||||||
|
numpassages++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
UP -> {
|
||||||
|
if @(celladdr(cx,cy-1)) & STONE == 0 {
|
||||||
|
@(celladdr(cx,cy)) |= UP
|
||||||
|
drawCell(cx,cy)
|
||||||
|
numpassages++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
DOWN -> {
|
||||||
|
if @(celladdr(cx,cy+1)) & STONE == 0 {
|
||||||
|
@(celladdr(cx,cy)) |= DOWN
|
||||||
|
drawCell(cx,cy)
|
||||||
|
numpassages++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} until numpassages==10
|
||||||
|
}
|
||||||
|
|
||||||
|
sub solve(uword max_time) -> bool {
|
||||||
|
ubyte cx = startCx
|
||||||
|
ubyte cy = startCy
|
||||||
|
const uword max_path_length = 1024
|
||||||
|
|
||||||
|
; the path through the maze can be longer than 256 so doesn't fit in a regular array.... :(
|
||||||
|
uword pathstack = memory("pathstack", max_path_length, 0)
|
||||||
|
uword pathstackptr = 0
|
||||||
|
|
||||||
|
@(celladdr(cx,cy)) |= WALKED
|
||||||
|
; txt.setcc(cx*2+1, cy*2+1, 81, 1)
|
||||||
|
|
||||||
|
while cbm.RDTIM16()<max_time {
|
||||||
|
solve_loop:
|
||||||
|
if cx==finishCx and cy==finishCy {
|
||||||
|
;txt.home()
|
||||||
|
txt.print("found! path length: ")
|
||||||
|
txt.print_uw(pathstackptr)
|
||||||
|
txt.nl()
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
ubyte cell = @(celladdr(cx,cy))
|
||||||
|
if cell & UP!=0 and @(celladdr(cx,cy-1)) & (WALKED|BACKTRACKED) ==0 {
|
||||||
|
@(pathstack + pathstackptr) = UP
|
||||||
|
;txt.setcc(cx*2+1, cy*2, 81, 3)
|
||||||
|
cy--
|
||||||
|
}
|
||||||
|
else if cell & DOWN !=0 and @(celladdr(cx,cy+1)) & (WALKED|BACKTRACKED) ==0 {
|
||||||
|
@(pathstack + pathstackptr) = DOWN
|
||||||
|
;txt.setcc(cx*2+1, cy*2+2, 81, 3)
|
||||||
|
cy++
|
||||||
|
}
|
||||||
|
else if cell & LEFT !=0 and @(celladdr(cx-1,cy)) & (WALKED|BACKTRACKED) ==0 {
|
||||||
|
@(pathstack + pathstackptr) = LEFT
|
||||||
|
;txt.setcc(cx*2, cy*2+1, 81, 3)
|
||||||
|
cx--
|
||||||
|
}
|
||||||
|
else if cell & RIGHT !=0 and @(celladdr(cx+1,cy)) & (WALKED|BACKTRACKED) ==0 {
|
||||||
|
@(pathstack + pathstackptr) = RIGHT
|
||||||
|
;txt.setcc(cx*2+2, cy*2+1, 81, 3)
|
||||||
|
cx++
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
; dead end, pop stack
|
||||||
|
pathstackptr--
|
||||||
|
if pathstackptr==65535 {
|
||||||
|
txt.print("no solution?!\n")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
@(celladdr(cx,cy)) |= BACKTRACKED
|
||||||
|
;txt.setcc(cx*2+1, cy*2+1, 81, 2)
|
||||||
|
when @(pathstack + pathstackptr) {
|
||||||
|
UP -> {
|
||||||
|
;txt.setcc(cx*2+1, cy*2+2, 81, 9)
|
||||||
|
cy++
|
||||||
|
}
|
||||||
|
DOWN -> {
|
||||||
|
;txt.setcc(cx*2+1, cy*2, 81, 9)
|
||||||
|
cy--
|
||||||
|
}
|
||||||
|
LEFT -> {
|
||||||
|
;txt.setcc(cx*2+2, cy*2+1, 81, 9)
|
||||||
|
cx++
|
||||||
|
}
|
||||||
|
RIGHT -> {
|
||||||
|
;txt.setcc(cx*2, cy*2+1, 81, 9)
|
||||||
|
cx--
|
||||||
|
|
||||||
|
score++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
goto solve_loop
|
||||||
|
}
|
||||||
|
pathstackptr++
|
||||||
|
if pathstackptr==max_path_length {
|
||||||
|
txt.print("stack overflow, path too long\n")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
@(celladdr(cx,cy)) |= WALKED
|
||||||
|
;txt.setcc(cx*2+1, cy*2+1, 81, 1)
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
sub celladdr(ubyte cx, ubyte cy) -> uword {
|
||||||
|
return cells+(numCellsHoriz as uword)*cy+cx
|
||||||
|
}
|
||||||
|
|
||||||
|
sub drawCell(ubyte cx, ubyte cy) {
|
||||||
|
return
|
||||||
|
; ubyte x = cx * 2 + 1
|
||||||
|
; ubyte y = cy * 2 + 1
|
||||||
|
; ubyte doors = @(celladdr(cx,cy))
|
||||||
|
; if doors & UP !=0
|
||||||
|
; txt.setcc(x, y-1, ' ', EMPTYCOLOR)
|
||||||
|
; if doors & RIGHT !=0
|
||||||
|
; txt.setcc(x+1, y, ' ', EMPTYCOLOR)
|
||||||
|
; if doors & DOWN !=0
|
||||||
|
; txt.setcc(x, y+1, ' ', EMPTYCOLOR)
|
||||||
|
; if doors & LEFT !=0
|
||||||
|
; txt.setcc(x-1, y, ' ', EMPTYCOLOR)
|
||||||
|
; if doors & STONE !=0
|
||||||
|
; txt.setcc(x, y, 160, WALLCOLOR)
|
||||||
|
; else
|
||||||
|
; txt.setcc(x, y, 32, EMPTYCOLOR)
|
||||||
|
;
|
||||||
|
; if doors & WALKED !=0
|
||||||
|
; txt.setcc(x, y, 81, 1)
|
||||||
|
; if doors & BACKTRACKED !=0
|
||||||
|
; txt.setcc(x, y, 81, 2)
|
||||||
|
}
|
||||||
|
|
||||||
|
sub initialize() {
|
||||||
|
sys.memset(cells, numCellsHoriz*numCellsVert, STONE)
|
||||||
|
; txt.fill_screen(160, WALLCOLOR)
|
||||||
|
drawStartFinish()
|
||||||
|
}
|
||||||
|
|
||||||
|
sub drawStartFinish() {
|
||||||
|
; txt.setcc(startCx*2+1,startCy*2+1,sc:'s',5)
|
||||||
|
; txt.setcc(finishCx*2+1, finishCy*2+1, sc:'f', 13)
|
||||||
|
}
|
||||||
|
}
|
63
benchmark-program/b_queens.p8
Normal file
63
benchmark-program/b_queens.p8
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
%import textio
|
||||||
|
|
||||||
|
; Recursive N-Queens solver.
|
||||||
|
; The problem is: find all possible ways to place 8 Queen chess pieces on a chess board, so that none of them attacks any other.
|
||||||
|
; (this program prints all solutions without taking mirroring and flipping the chess board into account)
|
||||||
|
; Note: this program can be compiled for multiple target systems.
|
||||||
|
|
||||||
|
queens {
|
||||||
|
const ubyte NUMQUEENS=8
|
||||||
|
ubyte[NUMQUEENS] board
|
||||||
|
|
||||||
|
sub could_place(ubyte row, ubyte col) -> bool {
|
||||||
|
if row==0
|
||||||
|
return true
|
||||||
|
ubyte i
|
||||||
|
for i in 0 to row-1 {
|
||||||
|
if board[i]==col or board[i]-i==col-row or board[i]+i==col+row
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
uword solution_count
|
||||||
|
uword maximum_duration
|
||||||
|
|
||||||
|
sub place_queen(ubyte row) -> bool {
|
||||||
|
if row == NUMQUEENS {
|
||||||
|
solution_count++
|
||||||
|
txt.chrout('.')
|
||||||
|
return cbm.RDTIM16()<maximum_duration
|
||||||
|
}
|
||||||
|
bool continue_running=true
|
||||||
|
ubyte col
|
||||||
|
for col in 0 to NUMQUEENS-1 {
|
||||||
|
if could_place(row, col) {
|
||||||
|
board[row] = col
|
||||||
|
; we need to save the local variables row and col.
|
||||||
|
sys.push(row)
|
||||||
|
sys.push(col)
|
||||||
|
continue_running = place_queen(row + 1)
|
||||||
|
; restore the local variables after the recursive call.
|
||||||
|
col = sys.pop()
|
||||||
|
row = sys.pop()
|
||||||
|
board[row] = 0
|
||||||
|
|
||||||
|
if not continue_running
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return continue_running
|
||||||
|
}
|
||||||
|
|
||||||
|
sub bench(uword max_time) -> uword {
|
||||||
|
solution_count = 0
|
||||||
|
maximum_duration = max_time
|
||||||
|
txt.nl()
|
||||||
|
cbm.SETTIM(0,0,0)
|
||||||
|
while cbm.RDTIM16() < maximum_duration {
|
||||||
|
void place_queen(0)
|
||||||
|
}
|
||||||
|
return solution_count
|
||||||
|
}
|
||||||
|
}
|
989
benchmark-program/b_textelite.p8
Normal file
989
benchmark-program/b_textelite.p8
Normal file
@ -0,0 +1,989 @@
|
|||||||
|
%import textio
|
||||||
|
%import conv
|
||||||
|
%import strings
|
||||||
|
|
||||||
|
|
||||||
|
textelite {
|
||||||
|
|
||||||
|
const ubyte numforLave = 7 ; Lave is 7th generated planet in galaxy one
|
||||||
|
const ubyte numforZaonce = 129
|
||||||
|
const ubyte numforDiso = 147
|
||||||
|
const ubyte numforRiedquat = 46
|
||||||
|
uword num_commands
|
||||||
|
|
||||||
|
sub bench(uword max_time) -> uword {
|
||||||
|
num_commands = 0
|
||||||
|
txt.lowercase()
|
||||||
|
cbm.SETTIM(0,0,0)
|
||||||
|
while cbm.RDTIM16()<max_time {
|
||||||
|
reinit()
|
||||||
|
run_commands(max_time)
|
||||||
|
}
|
||||||
|
return num_commands
|
||||||
|
}
|
||||||
|
|
||||||
|
sub reinit() {
|
||||||
|
;txt.clear_screen()
|
||||||
|
;txt.print("\n --- TextElite v1.3 ---\n")
|
||||||
|
txt.print("\nnew game\n")
|
||||||
|
elite_planet.set_seed(0, 0)
|
||||||
|
elite_galaxy.travel_to(1, numforLave)
|
||||||
|
elite_market.init(0) ; Lave's market is seeded with 0
|
||||||
|
elite_ship.init()
|
||||||
|
elite_planet.display(false, 0)
|
||||||
|
input_index = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
sub run_commands(uword max_time) {
|
||||||
|
while cbm.RDTIM16() < max_time {
|
||||||
|
str input = "????????"
|
||||||
|
;txt.print("\nCash: ")
|
||||||
|
;elite_util.print_10s(elite_ship.cash)
|
||||||
|
;txt.print("\nCommand (?=help): ")
|
||||||
|
ubyte num_chars = next_input(input)
|
||||||
|
;txt.nl()
|
||||||
|
if num_chars!=0 {
|
||||||
|
when input[0] {
|
||||||
|
'q' -> {
|
||||||
|
bool has_error = false
|
||||||
|
if elite_galaxy.number != 2 {
|
||||||
|
txt.print("\nERROR: galaxy is not 2: ")
|
||||||
|
txt.print_ub(elite_galaxy.number)
|
||||||
|
txt.nl()
|
||||||
|
has_error=true
|
||||||
|
}
|
||||||
|
if elite_planet.number != 164 {
|
||||||
|
txt.print("\nERROR: planet is not 164: ")
|
||||||
|
txt.print_ub(elite_planet.number)
|
||||||
|
txt.nl()
|
||||||
|
has_error=true
|
||||||
|
}
|
||||||
|
if elite_planet.x != 116 {
|
||||||
|
txt.print("\nERROR: planet.x is not 116: ")
|
||||||
|
txt.print_ub(elite_planet.x)
|
||||||
|
txt.nl()
|
||||||
|
has_error=true
|
||||||
|
}
|
||||||
|
if elite_planet.y != 201 {
|
||||||
|
txt.print("\nERROR: planet.y is not 201: ")
|
||||||
|
txt.print_ub(elite_planet.y)
|
||||||
|
txt.nl()
|
||||||
|
has_error=true
|
||||||
|
}
|
||||||
|
if "ribeen" != elite_planet.name {
|
||||||
|
txt.print("\nERROR: planet.name is not 'ribeen': ")
|
||||||
|
txt.print(elite_planet.name)
|
||||||
|
txt.nl()
|
||||||
|
has_error=true
|
||||||
|
}
|
||||||
|
if elite_ship.cash != 1212 {
|
||||||
|
txt.print("\nERROR: cash is not 1212: ")
|
||||||
|
txt.print_uw(elite_ship.cash)
|
||||||
|
txt.nl()
|
||||||
|
has_error=true
|
||||||
|
}
|
||||||
|
if elite_ship.fuel != 50 {
|
||||||
|
txt.print("\nERROR: fuel is not 50:")
|
||||||
|
txt.print_ub(elite_ship.fuel)
|
||||||
|
txt.nl()
|
||||||
|
has_error=true
|
||||||
|
}
|
||||||
|
if elite_ship.cargohold[0] != 3 {
|
||||||
|
txt.print("\nERROR: food is not 3:")
|
||||||
|
txt.print_ub(elite_ship.cargohold[0])
|
||||||
|
txt.nl()
|
||||||
|
has_error=true
|
||||||
|
}
|
||||||
|
if elite_ship.cargohold[1] != 0 {
|
||||||
|
txt.print("\nERROR: textiles is not 0:")
|
||||||
|
txt.print_ub(elite_ship.cargohold[1])
|
||||||
|
txt.nl()
|
||||||
|
has_error=true
|
||||||
|
}
|
||||||
|
if has_error
|
||||||
|
sys.exit(1)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
'b' -> elite_trader.do_buy()
|
||||||
|
's' -> elite_trader.do_sell()
|
||||||
|
'f' -> elite_trader.do_fuel()
|
||||||
|
'j' -> elite_trader.do_jump()
|
||||||
|
't' -> elite_trader.do_teleport()
|
||||||
|
'g' -> elite_trader.do_next_galaxy()
|
||||||
|
'i' -> elite_trader.do_info()
|
||||||
|
'm' -> {
|
||||||
|
if input[1]=='a' and input[2]=='p'
|
||||||
|
elite_trader.do_map()
|
||||||
|
else
|
||||||
|
elite_trader.do_show_market()
|
||||||
|
}
|
||||||
|
'l' -> elite_trader.do_local()
|
||||||
|
'c' -> elite_trader.do_cash()
|
||||||
|
'h' -> elite_trader.do_hold()
|
||||||
|
}
|
||||||
|
num_commands++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
str[] inputs = [
|
||||||
|
"i",
|
||||||
|
"diso",
|
||||||
|
"i",
|
||||||
|
"lave",
|
||||||
|
"m",
|
||||||
|
"b",
|
||||||
|
"food",
|
||||||
|
"15",
|
||||||
|
"map",
|
||||||
|
"g",
|
||||||
|
"map",
|
||||||
|
"l",
|
||||||
|
"j",
|
||||||
|
"zao",
|
||||||
|
"s",
|
||||||
|
"food",
|
||||||
|
"12",
|
||||||
|
"tele",
|
||||||
|
"quti",
|
||||||
|
"tele",
|
||||||
|
"aro",
|
||||||
|
"i",
|
||||||
|
"diso",
|
||||||
|
"i",
|
||||||
|
"lave",
|
||||||
|
"i",
|
||||||
|
"zao",
|
||||||
|
"galhyp",
|
||||||
|
"fuel",
|
||||||
|
"20",
|
||||||
|
"j",
|
||||||
|
"rib",
|
||||||
|
"i",
|
||||||
|
"rib",
|
||||||
|
"i",
|
||||||
|
"tiri",
|
||||||
|
"q",
|
||||||
|
0
|
||||||
|
]
|
||||||
|
|
||||||
|
ubyte input_index
|
||||||
|
|
||||||
|
sub next_input(str buffer) -> ubyte {
|
||||||
|
input_index++
|
||||||
|
return strings.copy(inputs[input_index], buffer)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
elite_trader {
|
||||||
|
str input = "??????????"
|
||||||
|
ubyte num_chars
|
||||||
|
|
||||||
|
sub do_jump() {
|
||||||
|
;txt.print("\nJump to what system? ")
|
||||||
|
jump_to_system()
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_teleport() {
|
||||||
|
;txt.print("\nCheat! Teleport to what system? ")
|
||||||
|
ubyte fuel = elite_ship.fuel
|
||||||
|
elite_ship.fuel = 255
|
||||||
|
jump_to_system()
|
||||||
|
elite_ship.fuel = fuel
|
||||||
|
}
|
||||||
|
|
||||||
|
sub jump_to_system() {
|
||||||
|
void textelite.next_input(input)
|
||||||
|
ubyte current_planet = elite_planet.number
|
||||||
|
ubyte x = elite_planet.x
|
||||||
|
ubyte y = elite_planet.y
|
||||||
|
if elite_galaxy.search_closest_planet(input) {
|
||||||
|
ubyte distance = elite_planet.distance(x, y)
|
||||||
|
if distance <= elite_ship.fuel {
|
||||||
|
elite_galaxy.init_market_for_planet()
|
||||||
|
elite_ship.fuel -= distance
|
||||||
|
;txt.print("\n\nHyperspace jump! Arrived at:\n")
|
||||||
|
elite_planet.display(true,0 )
|
||||||
|
return
|
||||||
|
}
|
||||||
|
;txt.print("\nInsufficient fuel\n")
|
||||||
|
} else {
|
||||||
|
;txt.print(" Not found!\n")
|
||||||
|
}
|
||||||
|
elite_galaxy.travel_to(elite_galaxy.number, current_planet)
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_buy() {
|
||||||
|
;txt.print("\nBuy what commodity? ")
|
||||||
|
str commodity = "???????????????"
|
||||||
|
void textelite.next_input(commodity)
|
||||||
|
ubyte ci = elite_market.match(commodity)
|
||||||
|
if ci & 128 !=0 {
|
||||||
|
txt.print("Unknown\n")
|
||||||
|
} else {
|
||||||
|
;txt.print("\nHow much? ")
|
||||||
|
void textelite.next_input(input)
|
||||||
|
ubyte amount = conv.str2ubyte(input)
|
||||||
|
if elite_market.current_quantity[ci] < amount {
|
||||||
|
txt.print(" Insufficient supply!\n")
|
||||||
|
} else {
|
||||||
|
uword price = elite_market.current_price[ci] * amount
|
||||||
|
;txt.print(" Total price: ")
|
||||||
|
;elite_util.print_10s(price)
|
||||||
|
if price > elite_ship.cash {
|
||||||
|
txt.print(" Not enough cash!\n")
|
||||||
|
} else {
|
||||||
|
elite_ship.cash -= price
|
||||||
|
elite_ship.cargohold[ci] += amount
|
||||||
|
elite_market.current_quantity[ci] -= amount
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_sell() {
|
||||||
|
;txt.print("\nSell what commodity? ")
|
||||||
|
str commodity = "???????????????"
|
||||||
|
void textelite.next_input(commodity)
|
||||||
|
ubyte ci = elite_market.match(commodity)
|
||||||
|
if ci & 128 !=0 {
|
||||||
|
txt.print("Unknown\n")
|
||||||
|
} else {
|
||||||
|
;txt.print("\nHow much? ")
|
||||||
|
void textelite.next_input(input)
|
||||||
|
ubyte amount = conv.str2ubyte(input)
|
||||||
|
if elite_ship.cargohold[ci] < amount {
|
||||||
|
txt.print(" Insufficient supply!\n")
|
||||||
|
} else {
|
||||||
|
uword price = elite_market.current_price[ci] * amount
|
||||||
|
;txt.print(" Total price: ")
|
||||||
|
;elite_util.print_10s(price)
|
||||||
|
elite_ship.cash += price
|
||||||
|
elite_ship.cargohold[ci] -= amount
|
||||||
|
elite_market.current_quantity[ci] += amount
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_fuel() {
|
||||||
|
;txt.print("\nBuy fuel. Amount? ")
|
||||||
|
void textelite.next_input(input)
|
||||||
|
ubyte buy_fuel = 10*conv.str2ubyte(input)
|
||||||
|
ubyte max_fuel = elite_ship.Max_fuel - elite_ship.fuel
|
||||||
|
if buy_fuel > max_fuel
|
||||||
|
buy_fuel = max_fuel
|
||||||
|
uword price = buy_fuel as uword * elite_ship.Fuel_cost
|
||||||
|
if price > elite_ship.cash {
|
||||||
|
txt.print("Not enough cash!\n")
|
||||||
|
} else {
|
||||||
|
elite_ship.cash -= price
|
||||||
|
elite_ship.fuel += buy_fuel
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_cash() {
|
||||||
|
;txt.print("\nCheat! Set cash amount: ")
|
||||||
|
void textelite.next_input(input)
|
||||||
|
elite_ship.cash = conv.str2uword(input)
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_hold() {
|
||||||
|
;txt.print("\nCheat! Set cargohold size: ")
|
||||||
|
void textelite.next_input(input)
|
||||||
|
elite_ship.Max_cargo = conv.str2ubyte(input)
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_next_galaxy() {
|
||||||
|
txt.print("\n>>>>> Galaxy Hyperjump!\n")
|
||||||
|
elite_galaxy.travel_to(elite_galaxy.number+1, elite_planet.number)
|
||||||
|
elite_planet.display(false, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_info() {
|
||||||
|
;txt.print("\nSystem name (empty=current): ")
|
||||||
|
num_chars = textelite.next_input(input)
|
||||||
|
if num_chars!=0 {
|
||||||
|
ubyte current_planet = elite_planet.number
|
||||||
|
ubyte x = elite_planet.x
|
||||||
|
ubyte y = elite_planet.y
|
||||||
|
if elite_galaxy.search_closest_planet(input) {
|
||||||
|
ubyte distance = elite_planet.distance(x, y)
|
||||||
|
elite_planet.display(false, distance)
|
||||||
|
} else {
|
||||||
|
;txt.print(" Not found!")
|
||||||
|
}
|
||||||
|
elite_galaxy.travel_to(elite_galaxy.number, current_planet)
|
||||||
|
} else {
|
||||||
|
elite_planet.display(false, 0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_local() {
|
||||||
|
elite_galaxy.local_area()
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_map() {
|
||||||
|
;txt.print("\n(l)ocal or (g)alaxy starmap? ")
|
||||||
|
num_chars = textelite.next_input(input)
|
||||||
|
if num_chars!=0 {
|
||||||
|
elite_galaxy.starmap(input[0]=='l')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub do_show_market() {
|
||||||
|
elite_market.display()
|
||||||
|
;txt.print("\nFuel: ")
|
||||||
|
;elite_util.print_10s(elite_ship.fuel)
|
||||||
|
;txt.print(" Cargohold space: ")
|
||||||
|
;txt.print_ub(elite_ship.cargo_free())
|
||||||
|
;txt.print("t\n")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
elite_ship {
|
||||||
|
const ubyte Max_fuel = 70
|
||||||
|
const ubyte Fuel_cost = 2
|
||||||
|
ubyte Max_cargo = 20
|
||||||
|
|
||||||
|
ubyte fuel
|
||||||
|
uword cash
|
||||||
|
ubyte[17] cargohold
|
||||||
|
|
||||||
|
sub init() {
|
||||||
|
sys.memset(cargohold, len(cargohold), 0)
|
||||||
|
fuel = Max_fuel
|
||||||
|
cash = 1000
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
elite_market {
|
||||||
|
ubyte[17] baseprices = [$13, $14, $41, $28, $53, $C4, $EB, $9A, $75, $4E, $7C, $B0, $20, $61, $AB, $2D, $35]
|
||||||
|
byte[17] gradients = [-$02, -$01, -$03, -$05, -$05, $08, $1D, $0E, $06, $01, $0d, -$09, -$01, -$01, -$02, -$01, $0F]
|
||||||
|
ubyte[17] basequants = [$06, $0A, $02, $E2, $FB, $36, $08, $38, $28, $11, $1D, $DC, $35, $42, $37, $FA, $C0]
|
||||||
|
ubyte[17] maskbytes = [$01, $03, $07, $1F, $0F, $03, $78, $03, $07, $1F, $07, $3F, $03, $07, $1F, $0F, $07]
|
||||||
|
str[17] names = ["Food", "Textiles", "Radioactives", "Slaves", "Liquor/Wines", "Luxuries", "Narcotics", "Computers",
|
||||||
|
"Machinery", "Alloys", "Firearms", "Furs", "Minerals", "Gold", "Platinum", "Gem-Stones", "Alien Items"]
|
||||||
|
|
||||||
|
ubyte[17] current_quantity
|
||||||
|
uword[17] current_price
|
||||||
|
|
||||||
|
sub init(ubyte fluct) {
|
||||||
|
; Prices and availabilities are influenced by the planet's economy type
|
||||||
|
; (0-7) and a random "fluctuation" byte that was kept within the saved
|
||||||
|
; commander position to keep the market prices constant over gamesaves.
|
||||||
|
; Availabilities must be saved with the game since the player alters them
|
||||||
|
; by buying (and selling(?))
|
||||||
|
;
|
||||||
|
; Almost all commands are one byte only and overflow "errors" are
|
||||||
|
; extremely frequent and exploited.
|
||||||
|
;
|
||||||
|
; Trade Item prices are held internally in a single byte=true value/4.
|
||||||
|
; The decimal point in prices is introduced only when printing them.
|
||||||
|
; Internally, all prices are integers.
|
||||||
|
; The player's cash is held in four bytes.
|
||||||
|
ubyte ci
|
||||||
|
for ci in 0 to len(names)-1 {
|
||||||
|
word product
|
||||||
|
byte changing
|
||||||
|
product = elite_planet.economy as word * gradients[ci]
|
||||||
|
changing = fluct & maskbytes[ci] as byte
|
||||||
|
ubyte q = (basequants[ci] as word + changing - product) as ubyte
|
||||||
|
if q & $80 !=0
|
||||||
|
q = 0 ; clip to positive 8-bit
|
||||||
|
current_quantity[ci] = q & $3f
|
||||||
|
q = (baseprices[ci] + changing + product) as ubyte
|
||||||
|
current_price[ci] = q * $0004
|
||||||
|
}
|
||||||
|
current_quantity[16] = 0 ; force nonavailability of Alien Items
|
||||||
|
}
|
||||||
|
|
||||||
|
sub display() {
|
||||||
|
return
|
||||||
|
; ubyte ci
|
||||||
|
; txt.nl()
|
||||||
|
; elite_planet.print_name_uppercase()
|
||||||
|
; txt.print(" trade market:\n COMMODITY / PRICE / AVAIL / IN HOLD\n")
|
||||||
|
; for ci in 0 to len(names)-1 {
|
||||||
|
; elite_util.print_right(13, names[ci])
|
||||||
|
; txt.print(" ")
|
||||||
|
; elite_util.print_10s(current_price[ci])
|
||||||
|
; txt.column(24)
|
||||||
|
; txt.print_ub(current_quantity[ci])
|
||||||
|
; txt.chrout(' ')
|
||||||
|
; when units[ci] {
|
||||||
|
; 0 -> txt.chrout('t')
|
||||||
|
; 1 -> txt.print("kg")
|
||||||
|
; 2 -> txt.chrout('g')
|
||||||
|
; }
|
||||||
|
; txt.column(32)
|
||||||
|
; txt.print_ub(elite_ship.cargohold[ci])
|
||||||
|
; txt.nl()
|
||||||
|
; }
|
||||||
|
}
|
||||||
|
|
||||||
|
sub match(uword nameptr) -> ubyte {
|
||||||
|
ubyte ci
|
||||||
|
for ci in 0 to len(names)-1 {
|
||||||
|
if elite_util.prefix_matches(nameptr, names[ci])
|
||||||
|
return ci
|
||||||
|
}
|
||||||
|
return 255
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
elite_galaxy {
|
||||||
|
const uword GALSIZE = 256
|
||||||
|
const uword base0 = $5A4A ; seeds for the first galaxy
|
||||||
|
const uword base1 = $0248
|
||||||
|
const uword base2 = $B753
|
||||||
|
|
||||||
|
str pn_pairs = "..lexegezacebisousesarmaindirea.eratenberalavetiedorquanteisrion"
|
||||||
|
|
||||||
|
ubyte number
|
||||||
|
|
||||||
|
uword[3] seed
|
||||||
|
|
||||||
|
sub init(ubyte galaxynum) {
|
||||||
|
number = 1
|
||||||
|
elite_planet.number = 255
|
||||||
|
seed[0] = base0
|
||||||
|
seed[1] = base1
|
||||||
|
seed[2] = base2
|
||||||
|
repeat galaxynum-1 {
|
||||||
|
nextgalaxy()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub nextgalaxy() {
|
||||||
|
textelite.num_commands++
|
||||||
|
|
||||||
|
seed[0] = twist(seed[0])
|
||||||
|
seed[1] = twist(seed[1])
|
||||||
|
seed[2] = twist(seed[2])
|
||||||
|
number++
|
||||||
|
if number==9
|
||||||
|
number = 1
|
||||||
|
}
|
||||||
|
|
||||||
|
sub travel_to(ubyte galaxynum, ubyte system) {
|
||||||
|
init(galaxynum)
|
||||||
|
generate_next_planet() ; always at least planet 0 (separate to avoid repeat ubyte overflow)
|
||||||
|
repeat system {
|
||||||
|
generate_next_planet()
|
||||||
|
textelite.num_commands++
|
||||||
|
}
|
||||||
|
elite_planet.name = make_current_planet_name()
|
||||||
|
init_market_for_planet()
|
||||||
|
}
|
||||||
|
|
||||||
|
sub init_market_for_planet() {
|
||||||
|
elite_market.init(lsb(seed[0])+msb(seed[2]))
|
||||||
|
}
|
||||||
|
|
||||||
|
sub search_closest_planet(uword nameptr) -> bool {
|
||||||
|
textelite.num_commands++
|
||||||
|
|
||||||
|
ubyte x = elite_planet.x
|
||||||
|
ubyte y = elite_planet.y
|
||||||
|
ubyte current_planet_num = elite_planet.number
|
||||||
|
|
||||||
|
init(number)
|
||||||
|
bool found = false
|
||||||
|
ubyte current_closest_pi
|
||||||
|
ubyte current_distance = 127
|
||||||
|
ubyte pi
|
||||||
|
for pi in 0 to 255 {
|
||||||
|
generate_next_planet()
|
||||||
|
elite_planet.name = make_current_planet_name()
|
||||||
|
if elite_util.prefix_matches(nameptr, elite_planet.name) {
|
||||||
|
ubyte distance = elite_planet.distance(x, y)
|
||||||
|
if distance < current_distance {
|
||||||
|
current_distance = distance
|
||||||
|
current_closest_pi = pi
|
||||||
|
found = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if found
|
||||||
|
travel_to(number, current_closest_pi)
|
||||||
|
else
|
||||||
|
travel_to(number, current_planet_num)
|
||||||
|
|
||||||
|
return found
|
||||||
|
}
|
||||||
|
|
||||||
|
sub local_area() {
|
||||||
|
ubyte current_planet = elite_planet.number
|
||||||
|
ubyte px = elite_planet.x
|
||||||
|
ubyte py = elite_planet.y
|
||||||
|
ubyte pn = 0
|
||||||
|
|
||||||
|
init(number)
|
||||||
|
; txt.print("\nGalaxy #")
|
||||||
|
; txt.print_ub(number)
|
||||||
|
; txt.print(" - systems in vicinity:\n")
|
||||||
|
do {
|
||||||
|
generate_next_planet()
|
||||||
|
ubyte distance = elite_planet.distance(px, py)
|
||||||
|
if distance <= elite_ship.Max_fuel {
|
||||||
|
; if distance <= elite_ship.fuel
|
||||||
|
; txt.chrout('*')
|
||||||
|
; else
|
||||||
|
; txt.chrout('-')
|
||||||
|
; txt.spc()
|
||||||
|
elite_planet.name = make_current_planet_name()
|
||||||
|
elite_planet.display(true, distance)
|
||||||
|
}
|
||||||
|
pn++
|
||||||
|
} until pn==0
|
||||||
|
|
||||||
|
travel_to(number, current_planet)
|
||||||
|
}
|
||||||
|
|
||||||
|
sub starmap(bool local) {
|
||||||
|
ubyte current_planet = elite_planet.number
|
||||||
|
ubyte px = elite_planet.x
|
||||||
|
ubyte py = elite_planet.y
|
||||||
|
str current_name = " " ; 8 max
|
||||||
|
ubyte pn = 0
|
||||||
|
|
||||||
|
current_name = elite_planet.name
|
||||||
|
init(number)
|
||||||
|
; txt.clear_screen()
|
||||||
|
; txt.print("Galaxy #")
|
||||||
|
; txt.print_ub(number)
|
||||||
|
; if local
|
||||||
|
; txt.print(" - local systems")
|
||||||
|
; else
|
||||||
|
; txt.print(" - galaxy")
|
||||||
|
; txt.print(" starmap:\n")
|
||||||
|
ubyte max_distance = 255
|
||||||
|
if local
|
||||||
|
max_distance = elite_ship.Max_fuel
|
||||||
|
ubyte home_sx
|
||||||
|
ubyte home_sy
|
||||||
|
ubyte home_distance
|
||||||
|
|
||||||
|
do {
|
||||||
|
generate_next_planet()
|
||||||
|
ubyte distance = elite_planet.distance(px, py)
|
||||||
|
if distance <= max_distance {
|
||||||
|
elite_planet.name = make_current_planet_name()
|
||||||
|
elite_planet.name[0] = strings.upperchar(elite_planet.name[0])
|
||||||
|
uword tx = elite_planet.x
|
||||||
|
uword ty = elite_planet.y
|
||||||
|
if local {
|
||||||
|
tx = tx + 24 - px
|
||||||
|
ty = ty + 24 - py
|
||||||
|
}
|
||||||
|
ubyte sx = display_scale_x(tx)
|
||||||
|
ubyte sy = display_scale_y(ty)
|
||||||
|
ubyte char = '*'
|
||||||
|
if elite_planet.number==current_planet
|
||||||
|
char = '%'
|
||||||
|
if local {
|
||||||
|
print_planet_details(elite_planet.name, sx, sy, distance)
|
||||||
|
} else if elite_planet.number==current_planet {
|
||||||
|
home_distance = distance
|
||||||
|
home_sx = sx
|
||||||
|
home_sy = sy
|
||||||
|
}
|
||||||
|
; txt.setchr(2+sx, 2+sy, char)
|
||||||
|
}
|
||||||
|
pn++
|
||||||
|
} until pn==0
|
||||||
|
|
||||||
|
if not local
|
||||||
|
print_planet_details(current_name, home_sx, home_sy, home_distance)
|
||||||
|
|
||||||
|
; if local
|
||||||
|
; txt.plot(0, display_scale_y(64) + 4)
|
||||||
|
; else
|
||||||
|
; txt.plot(0, display_scale_y(256) + 4 as ubyte)
|
||||||
|
travel_to(number, current_planet)
|
||||||
|
|
||||||
|
sub print_planet_details(str name, ubyte screenx, ubyte screeny, ubyte d) {
|
||||||
|
return
|
||||||
|
; txt.plot(2+screenx-2, 2+screeny+1)
|
||||||
|
; txt.print(name)
|
||||||
|
; if d!=0 {
|
||||||
|
; txt.plot(2+screenx-2, 2+screeny+2)
|
||||||
|
; elite_util.print_10s(d)
|
||||||
|
; txt.print(" LY")
|
||||||
|
; }
|
||||||
|
}
|
||||||
|
|
||||||
|
sub display_scale_x(uword x) -> ubyte {
|
||||||
|
if local
|
||||||
|
return x/2 as ubyte
|
||||||
|
return x/8 as ubyte
|
||||||
|
}
|
||||||
|
|
||||||
|
sub display_scale_y(uword y) -> ubyte {
|
||||||
|
if local
|
||||||
|
return y/4 as ubyte
|
||||||
|
return y/16 as ubyte
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ubyte pn_pair1
|
||||||
|
ubyte pn_pair2
|
||||||
|
ubyte pn_pair3
|
||||||
|
ubyte pn_pair4
|
||||||
|
bool longname
|
||||||
|
|
||||||
|
sub generate_next_planet() {
|
||||||
|
determine_planet_properties()
|
||||||
|
longname = lsb(seed[0]) & 64 !=0
|
||||||
|
|
||||||
|
; Always four iterations of random number
|
||||||
|
pn_pair1 = (msb(seed[2]) & 31) * 2
|
||||||
|
tweakseed()
|
||||||
|
pn_pair2 = (msb(seed[2]) & 31) * 2
|
||||||
|
tweakseed()
|
||||||
|
pn_pair3 = (msb(seed[2]) & 31) * 2
|
||||||
|
tweakseed()
|
||||||
|
pn_pair4 = (msb(seed[2]) & 31) * 2
|
||||||
|
tweakseed()
|
||||||
|
}
|
||||||
|
|
||||||
|
sub make_current_planet_name() -> str {
|
||||||
|
ubyte ni = 0
|
||||||
|
str name = " " ; max 8
|
||||||
|
|
||||||
|
if pn_pairs[pn_pair1] != '.' {
|
||||||
|
name[ni] = pn_pairs[pn_pair1]
|
||||||
|
ni++
|
||||||
|
}
|
||||||
|
if pn_pairs[pn_pair1+1] != '.' {
|
||||||
|
name[ni] = pn_pairs[pn_pair1+1]
|
||||||
|
ni++
|
||||||
|
}
|
||||||
|
if pn_pairs[pn_pair2] != '.' {
|
||||||
|
name[ni] = pn_pairs[pn_pair2]
|
||||||
|
ni++
|
||||||
|
}
|
||||||
|
if pn_pairs[pn_pair2+1] != '.' {
|
||||||
|
name[ni] = pn_pairs[pn_pair2+1]
|
||||||
|
ni++
|
||||||
|
}
|
||||||
|
if pn_pairs[pn_pair3] != '.' {
|
||||||
|
name[ni] = pn_pairs[pn_pair3]
|
||||||
|
ni++
|
||||||
|
}
|
||||||
|
if pn_pairs[pn_pair3+1] != '.' {
|
||||||
|
name[ni] = pn_pairs[pn_pair3+1]
|
||||||
|
ni++
|
||||||
|
}
|
||||||
|
|
||||||
|
if longname {
|
||||||
|
if pn_pairs[pn_pair4] != '.' {
|
||||||
|
name[ni] = pn_pairs[pn_pair4]
|
||||||
|
ni++
|
||||||
|
}
|
||||||
|
if pn_pairs[pn_pair4+1] != '.' {
|
||||||
|
name[ni] = pn_pairs[pn_pair4+1]
|
||||||
|
ni++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
name[ni] = 0
|
||||||
|
return name
|
||||||
|
}
|
||||||
|
|
||||||
|
sub determine_planet_properties() {
|
||||||
|
; create the planet's characteristics
|
||||||
|
elite_planet.number++
|
||||||
|
elite_planet.x = msb(seed[1])
|
||||||
|
elite_planet.y = msb(seed[0])
|
||||||
|
elite_planet.govtype = lsb(seed[1]) >> 3 & 7 ; bits 3,4 &5 of w1
|
||||||
|
elite_planet.economy = msb(seed[0]) & 7 ; bits 8,9 &A of w0
|
||||||
|
if elite_planet.govtype <= 1
|
||||||
|
elite_planet.economy = (elite_planet.economy | 2)
|
||||||
|
elite_planet.techlevel = (msb(seed[1]) & 3) + (elite_planet.economy ^ 7)
|
||||||
|
elite_planet.techlevel += elite_planet.govtype >> 1
|
||||||
|
if elite_planet.govtype & 1 !=0
|
||||||
|
elite_planet.techlevel++
|
||||||
|
elite_planet.population = 4 * elite_planet.techlevel + elite_planet.economy
|
||||||
|
elite_planet.population += elite_planet.govtype + 1
|
||||||
|
elite_planet.productivity = ((elite_planet.economy ^ 7) + 3) * (elite_planet.govtype + 4)
|
||||||
|
elite_planet.productivity *= elite_planet.population * 8
|
||||||
|
ubyte seed2_msb = msb(seed[2])
|
||||||
|
elite_planet.radius = mkword((seed2_msb & 15) + 11, elite_planet.x)
|
||||||
|
elite_planet.species_is_alien = lsb(seed[2]) & 128 !=0 ; bit 7 of w2_lo
|
||||||
|
if elite_planet.species_is_alien {
|
||||||
|
elite_planet.species_size = (seed2_msb >> 2) & 7 ; bits 2-4 of w2_hi
|
||||||
|
elite_planet.species_color = seed2_msb >> 5 ; bits 5-7 of w2_hi
|
||||||
|
elite_planet.species_look = (seed2_msb ^ msb(seed[1])) & 7 ;bits 0-2 of (w0_hi EOR w1_hi)
|
||||||
|
elite_planet.species_kind = (elite_planet.species_look + (seed2_msb & 3)) & 7 ;Add bits 0-1 of w2_hi to A from previous step, and take bits 0-2 of the result
|
||||||
|
}
|
||||||
|
|
||||||
|
elite_planet.goatsoup_seed[0] = lsb(seed[1])
|
||||||
|
elite_planet.goatsoup_seed[1] = msb(seed[1])
|
||||||
|
elite_planet.goatsoup_seed[2] = lsb(seed[2])
|
||||||
|
elite_planet.goatsoup_seed[3] = seed2_msb
|
||||||
|
}
|
||||||
|
|
||||||
|
sub tweakseed() {
|
||||||
|
uword temp = seed[0] + seed[1] + seed[2]
|
||||||
|
seed[0] = seed[1]
|
||||||
|
seed[1] = seed[2]
|
||||||
|
seed[2] = temp
|
||||||
|
}
|
||||||
|
|
||||||
|
sub twist(uword x) -> uword {
|
||||||
|
ubyte xh = msb(x)
|
||||||
|
ubyte xl = lsb(x)
|
||||||
|
xh <<= 1 ; make sure carry flag is not used on first shift!
|
||||||
|
rol(xl)
|
||||||
|
return mkword(xh, xl)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
elite_planet {
|
||||||
|
str[] @nosplit words81 = ["fabled", "notable", "well known", "famous", "noted"]
|
||||||
|
str[] @nosplit words82 = ["very", "mildly", "most", "reasonably", ""]
|
||||||
|
str[] @nosplit words83 = ["ancient", "\x95", "great", "vast", "pink"]
|
||||||
|
str[] @nosplit words84 = ["\x9E \x9D plantations", "mountains", "\x9C", "\x94 forests", "oceans"]
|
||||||
|
str[] @nosplit words85 = ["shyness", "silliness", "mating traditions", "loathing of \x86", "love for \x86"]
|
||||||
|
str[] @nosplit words86 = ["food blenders", "tourists", "poetry", "discos", "\x8E"]
|
||||||
|
str[] @nosplit words87 = ["talking tree", "crab", "bat", "lobst", "\xB2"]
|
||||||
|
str[] @nosplit words88 = ["beset", "plagued", "ravaged", "cursed", "scourged"]
|
||||||
|
str[] @nosplit words89 = ["\x96 civil war", "\x9B \x98 \x99s", "a \x9B disease", "\x96 earthquakes", "\x96 solar activity"]
|
||||||
|
str[] @nosplit words8A = ["its \x83 \x84", "the \xB1 \x98 \x99", "its inhabitants' \x9A \x85", "\xA1", "its \x8D \x8E"]
|
||||||
|
str[] @nosplit words8B = ["juice", "brandy", "water", "brew", "gargle blasters"]
|
||||||
|
str[] @nosplit words8C = ["\xB2", "\xB1 \x99", "\xB1 \xB2", "\xB1 \x9B", "\x9B \xB2"]
|
||||||
|
str[] @nosplit words8D = ["fabulous", "exotic", "hoopy", "unusual", "exciting"]
|
||||||
|
str[] @nosplit words8E = ["cuisine", "night life", "casinos", "sit coms", " \xA1 "]
|
||||||
|
str[] @nosplit words8F = ["\xB0", "The planet \xB0", "The world \xB0", "This planet", "This world"]
|
||||||
|
str[] @nosplit words90 = ["n unremarkable", " boring", " dull", " tedious", " revolting"]
|
||||||
|
str[] @nosplit words91 = ["planet", "world", "place", "little planet", "dump"]
|
||||||
|
str[] @nosplit words92 = ["wasp", "moth", "grub", "ant", "\xB2"]
|
||||||
|
str[] @nosplit words93 = ["poet", "arts graduate", "yak", "snail", "slug"]
|
||||||
|
str[] @nosplit words94 = ["tropical", "dense", "rain", "impenetrable", "exuberant"]
|
||||||
|
str[] @nosplit words95 = ["funny", "wierd", "unusual", "strange", "peculiar"]
|
||||||
|
str[] @nosplit words96 = ["frequent", "occasional", "unpredictable", "dreadful", "deadly"]
|
||||||
|
str[] @nosplit words97 = ["\x82 \x81 for \x8A", "\x82 \x81 for \x8A and \x8A", "\x88 by \x89", "\x82 \x81 for \x8A but \x88 by \x89", "a\x90 \x91"]
|
||||||
|
str[] @nosplit words98 = ["\x9B", "mountain", "edible", "tree", "spotted"]
|
||||||
|
str[] @nosplit words99 = ["\x9F", "\xA0", "\x87oid", "\x93", "\x92"]
|
||||||
|
str[] @nosplit words9A = ["ancient", "exceptional", "eccentric", "ingrained", "\x95"]
|
||||||
|
str[] @nosplit words9B = ["killer", "deadly", "evil", "lethal", "vicious"]
|
||||||
|
str[] @nosplit words9C = ["parking meters", "dust clouds", "ice bergs", "rock formations", "volcanoes"]
|
||||||
|
str[] @nosplit words9D = ["plant", "tulip", "banana", "corn", "\xB2weed"]
|
||||||
|
str[] @nosplit words9E = ["\xB2", "\xB1 \xB2", "\xB1 \x9B", "inhabitant", "\xB1 \xB2"]
|
||||||
|
str[] @nosplit words9F = ["shrew", "beast", "bison", "snake", "wolf"]
|
||||||
|
str[] @nosplit wordsA0 = ["leopard", "cat", "monkey", "goat", "fish"]
|
||||||
|
str[] @nosplit wordsA1 = ["\x8C \x8B", "\xB1 \x9F \xA2", "its \x8D \xA0 \xA2", "\xA3 \xA4", "\x8C \x8B"]
|
||||||
|
str[] @nosplit wordsA2 = ["meat", "cutlet", "steak", "burgers", "soup"]
|
||||||
|
str[] @nosplit wordsA3 = ["ice", "mud", "Zero-G", "vacuum", "\xB1 ultra"]
|
||||||
|
str[] @nosplit wordsA4 = ["hockey", "cricket", "karate", "polo", "tennis"]
|
||||||
|
|
||||||
|
uword[] @shared wordlists = [
|
||||||
|
words81, words82, words83, words84, words85, words86, words87, words88,
|
||||||
|
words89, words8A, words8B, words8C, words8D, words8E, words8F, words90,
|
||||||
|
words91, words92, words93, words94, words95, words96, words97, words98,
|
||||||
|
words99, words9A, words9B, words9C, words9D, words9E, words9F, wordsA0,
|
||||||
|
wordsA1, wordsA2, wordsA3, wordsA4]
|
||||||
|
|
||||||
|
str pairs0 = "abouseitiletstonlonuthnoallexegezacebisousesarmaindirea.eratenbe"
|
||||||
|
|
||||||
|
ubyte[4] goatsoup_rnd = [0, 0, 0, 0]
|
||||||
|
ubyte[4] goatsoup_seed = [0, 0, 0, 0]
|
||||||
|
|
||||||
|
str name = " " ; 8 max
|
||||||
|
ubyte number ; starts at 0 in new galaxy, then increases by 1 for each generated planet
|
||||||
|
ubyte x
|
||||||
|
ubyte y
|
||||||
|
ubyte economy
|
||||||
|
ubyte govtype
|
||||||
|
ubyte techlevel
|
||||||
|
ubyte population
|
||||||
|
uword productivity
|
||||||
|
uword radius
|
||||||
|
bool species_is_alien ; otherwise "Human Colonials"
|
||||||
|
ubyte species_size
|
||||||
|
ubyte species_color
|
||||||
|
ubyte species_look
|
||||||
|
ubyte species_kind
|
||||||
|
|
||||||
|
sub set_seed(uword s1, uword s2) {
|
||||||
|
goatsoup_seed[0] = lsb(s1)
|
||||||
|
goatsoup_seed[1] = msb(s1)
|
||||||
|
goatsoup_seed[2] = lsb(s2)
|
||||||
|
goatsoup_seed[3] = msb(s2)
|
||||||
|
reset_rnd()
|
||||||
|
}
|
||||||
|
|
||||||
|
sub reset_rnd() {
|
||||||
|
goatsoup_rnd[0] = goatsoup_seed[0]
|
||||||
|
goatsoup_rnd[1] = goatsoup_seed[1]
|
||||||
|
goatsoup_rnd[2] = goatsoup_seed[2]
|
||||||
|
goatsoup_rnd[3] = goatsoup_seed[3]
|
||||||
|
}
|
||||||
|
|
||||||
|
sub random_name() -> str {
|
||||||
|
ubyte ii
|
||||||
|
str randname = " " ; 8 chars max
|
||||||
|
ubyte nx = 0
|
||||||
|
for ii in 0 to goatsoup_rnd_number() & 3 {
|
||||||
|
ubyte xx = goatsoup_rnd_number() & $3e
|
||||||
|
if pairs0[xx] != '.' {
|
||||||
|
randname[nx] = pairs0[xx]
|
||||||
|
nx++
|
||||||
|
}
|
||||||
|
xx++
|
||||||
|
if pairs0[xx] != '.' {
|
||||||
|
randname[nx] = pairs0[xx]
|
||||||
|
nx++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
randname[nx] = 0
|
||||||
|
randname[0] = strings.upperchar(randname[0])
|
||||||
|
return randname
|
||||||
|
}
|
||||||
|
|
||||||
|
sub goatsoup_rnd_number() -> ubyte {
|
||||||
|
ubyte xx = goatsoup_rnd[0] * 2
|
||||||
|
uword a = xx as uword + goatsoup_rnd[2]
|
||||||
|
if goatsoup_rnd[0] > 127
|
||||||
|
a ++
|
||||||
|
goatsoup_rnd[0] = lsb(a)
|
||||||
|
goatsoup_rnd[2] = xx
|
||||||
|
xx = goatsoup_rnd[1]
|
||||||
|
ubyte ac = xx + goatsoup_rnd[3] + msb(a)
|
||||||
|
goatsoup_rnd[1] = ac
|
||||||
|
goatsoup_rnd[3] = xx
|
||||||
|
return ac
|
||||||
|
}
|
||||||
|
|
||||||
|
sub distance(ubyte px, ubyte py) -> ubyte {
|
||||||
|
uword ax
|
||||||
|
uword ay
|
||||||
|
if px>x
|
||||||
|
ax=px-x
|
||||||
|
else
|
||||||
|
ax=x-px
|
||||||
|
if py>y
|
||||||
|
ay=py-y
|
||||||
|
else
|
||||||
|
ay=y-py
|
||||||
|
ay /= 2
|
||||||
|
ubyte d = sqrt(ax*ax + ay*ay)
|
||||||
|
if d>63
|
||||||
|
return 255
|
||||||
|
return d*4
|
||||||
|
}
|
||||||
|
|
||||||
|
sub soup() -> str {
|
||||||
|
str planet_result = " " * 160
|
||||||
|
uword[6] source_stack
|
||||||
|
ubyte stack_ptr = 0
|
||||||
|
str start_source = "\x8F is \x97."
|
||||||
|
uword source_ptr = &start_source
|
||||||
|
uword result_ptr = &planet_result
|
||||||
|
|
||||||
|
reset_rnd()
|
||||||
|
recursive_soup()
|
||||||
|
return planet_result
|
||||||
|
|
||||||
|
sub recursive_soup() {
|
||||||
|
repeat {
|
||||||
|
ubyte c = @(source_ptr)
|
||||||
|
source_ptr++
|
||||||
|
if c == $00 {
|
||||||
|
@(result_ptr) = 0
|
||||||
|
return
|
||||||
|
}
|
||||||
|
else if c <= $80 {
|
||||||
|
@(result_ptr) = c
|
||||||
|
result_ptr++
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if c <= $a4 {
|
||||||
|
ubyte rnr = goatsoup_rnd_number()
|
||||||
|
ubyte wordNr = ((rnr >= $33) as ubyte) + ((rnr >= $66) as ubyte) + ((rnr >= $99) as ubyte) + ((rnr >= $CC) as ubyte)
|
||||||
|
source_stack[stack_ptr] = source_ptr
|
||||||
|
stack_ptr++
|
||||||
|
source_ptr = getword(c, wordNr)
|
||||||
|
recursive_soup() ; RECURSIVE CALL - ignore the warning message from the compiler; we don't use local variables or parameters so we're safe in this case
|
||||||
|
stack_ptr--
|
||||||
|
source_ptr = source_stack[stack_ptr]
|
||||||
|
} else {
|
||||||
|
if c == $b0 {
|
||||||
|
@(result_ptr) = strings.upperchar(name[0])
|
||||||
|
result_ptr++
|
||||||
|
concat_string(&name + 1)
|
||||||
|
}
|
||||||
|
else if c == $b1 {
|
||||||
|
@(result_ptr) = strings.upperchar(name[0])
|
||||||
|
result_ptr++
|
||||||
|
ubyte ni
|
||||||
|
for ni in 1 to len(name) {
|
||||||
|
ubyte cc = name[ni]
|
||||||
|
if cc in ['e', 'o', 0]
|
||||||
|
break
|
||||||
|
else {
|
||||||
|
@(result_ptr) = cc
|
||||||
|
result_ptr++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@(result_ptr) = 'i'
|
||||||
|
result_ptr++
|
||||||
|
@(result_ptr) = 'a'
|
||||||
|
result_ptr++
|
||||||
|
@(result_ptr) = 'n'
|
||||||
|
result_ptr++
|
||||||
|
}
|
||||||
|
else if c == $b2 {
|
||||||
|
concat_string(random_name())
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
@(result_ptr) = c
|
||||||
|
result_ptr++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub concat_string(uword str_ptr) {
|
||||||
|
repeat {
|
||||||
|
ubyte c = @(str_ptr)
|
||||||
|
if c==0
|
||||||
|
break
|
||||||
|
else {
|
||||||
|
@(result_ptr) = c
|
||||||
|
str_ptr++
|
||||||
|
result_ptr++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
sub display(bool compressed, ubyte distance) {
|
||||||
|
txt.print(soup())
|
||||||
|
txt.nl()
|
||||||
|
}
|
||||||
|
|
||||||
|
sub getword(ubyte listnum, ubyte wordidx) -> uword {
|
||||||
|
uword list = wordlists[listnum-$81]
|
||||||
|
return peekw(list + wordidx*2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
elite_util {
|
||||||
|
sub prefix_matches(uword prefixptr, uword stringptr) -> bool {
|
||||||
|
repeat {
|
||||||
|
ubyte pc = @(prefixptr)
|
||||||
|
ubyte sc = @(stringptr)
|
||||||
|
if pc == 0
|
||||||
|
return true
|
||||||
|
; to lowercase for case insensitive compare:
|
||||||
|
if strings.lowerchar(pc)!=strings.lowerchar(sc)
|
||||||
|
return false
|
||||||
|
prefixptr++
|
||||||
|
stringptr++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
112
benchmark-program/benchmark.p8
Normal file
112
benchmark-program/benchmark.p8
Normal file
@ -0,0 +1,112 @@
|
|||||||
|
|
||||||
|
; This benchmark program is meant to check for regressions in the
|
||||||
|
; Prog8 compiler's code-generator (performance wise).
|
||||||
|
;
|
||||||
|
; As the X16 computer is a more or less fixed system, it's not very useful
|
||||||
|
; to benchmark the computer itself with.
|
||||||
|
|
||||||
|
|
||||||
|
%import textio
|
||||||
|
%import b_adpcm
|
||||||
|
%import b_circles
|
||||||
|
%import b_3d
|
||||||
|
%import b_life
|
||||||
|
%import b_mandelbrot
|
||||||
|
%import b_queens
|
||||||
|
%import b_textelite
|
||||||
|
%import b_maze
|
||||||
|
|
||||||
|
%zeropage basicsafe
|
||||||
|
%option no_sysinit
|
||||||
|
|
||||||
|
|
||||||
|
main {
|
||||||
|
|
||||||
|
str[20] benchmark_names
|
||||||
|
uword[20] benchmark_score
|
||||||
|
|
||||||
|
|
||||||
|
sub start() {
|
||||||
|
ubyte benchmark_number
|
||||||
|
|
||||||
|
void cx16.set_screen_mode(3)
|
||||||
|
txt.color2(1, 6)
|
||||||
|
txt.clear_screen()
|
||||||
|
|
||||||
|
txt.print("\n\n\n prog8 compiler benchmark tests.\n")
|
||||||
|
sys.wait(60)
|
||||||
|
|
||||||
|
benchmark_number = 0
|
||||||
|
|
||||||
|
announce_benchmark("maze solver")
|
||||||
|
benchmark_score[benchmark_number] = maze.bench(300)
|
||||||
|
benchmark_number++
|
||||||
|
|
||||||
|
announce_benchmark("n-queens")
|
||||||
|
benchmark_score[benchmark_number] = queens.bench(300)
|
||||||
|
benchmark_number++
|
||||||
|
|
||||||
|
announce_benchmark("mandelbrot (floating point)")
|
||||||
|
benchmark_score[benchmark_number] = mandelbrot.calc(400)
|
||||||
|
benchmark_number++
|
||||||
|
|
||||||
|
announce_benchmark("game of life")
|
||||||
|
benchmark_score[benchmark_number] = life.benchmark(300)
|
||||||
|
benchmark_number++
|
||||||
|
|
||||||
|
announce_benchmark("3d model rotation")
|
||||||
|
benchmark_score[benchmark_number] = rotate3d.benchmark(300)
|
||||||
|
benchmark_number++
|
||||||
|
|
||||||
|
announce_benchmark("adpcm audio decoding")
|
||||||
|
benchmark_score[benchmark_number] = adpcm.decode_benchmark(300)
|
||||||
|
benchmark_number++
|
||||||
|
|
||||||
|
announce_benchmark("circles with gfx_lores")
|
||||||
|
benchmark_score[benchmark_number] = circles.draw(false, 300)
|
||||||
|
benchmark_number++
|
||||||
|
|
||||||
|
; announce_benchmark("circles with kernal")
|
||||||
|
; benchmark_score[benchmark_number] = circles.draw(true, 300)
|
||||||
|
; benchmark_number++
|
||||||
|
|
||||||
|
announce_benchmark("text-elite")
|
||||||
|
benchmark_score[benchmark_number] = textelite.bench(120)
|
||||||
|
benchmark_number++
|
||||||
|
|
||||||
|
benchmark_names[benchmark_number] = 0
|
||||||
|
benchmark_score[benchmark_number] = 0
|
||||||
|
|
||||||
|
void cx16.set_screen_mode(3)
|
||||||
|
txt.uppercase()
|
||||||
|
txt.color2(1, 6)
|
||||||
|
uword final_score
|
||||||
|
benchmark_number = 0
|
||||||
|
txt.print("\nscore benchmark\n\n")
|
||||||
|
do {
|
||||||
|
txt.spc()
|
||||||
|
txt.print_uw(benchmark_score[benchmark_number])
|
||||||
|
txt.column(6)
|
||||||
|
txt.print(benchmark_names[benchmark_number])
|
||||||
|
final_score += benchmark_score[benchmark_number]
|
||||||
|
txt.nl()
|
||||||
|
benchmark_number++
|
||||||
|
} until benchmark_names[benchmark_number]==0
|
||||||
|
|
||||||
|
txt.print("\n\nfinal score : ")
|
||||||
|
txt.print_uw(final_score)
|
||||||
|
txt.nl()
|
||||||
|
|
||||||
|
sub announce_benchmark(str name) {
|
||||||
|
benchmark_names[benchmark_number] = name
|
||||||
|
void cx16.set_screen_mode(3)
|
||||||
|
txt.uppercase()
|
||||||
|
txt.color2(1, 6)
|
||||||
|
txt.clear_screen()
|
||||||
|
txt.plot(4, 6)
|
||||||
|
txt.print(benchmark_names[benchmark_number])
|
||||||
|
txt.nl()
|
||||||
|
sys.wait(60)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
10
build.gradle
10
build.gradle
@ -1,10 +0,0 @@
|
|||||||
plugins {
|
|
||||||
id "org.jetbrains.kotlin.jvm" version "$kotlinVersion" apply false
|
|
||||||
}
|
|
||||||
|
|
||||||
allprojects {
|
|
||||||
repositories {
|
|
||||||
mavenLocal()
|
|
||||||
mavenCentral()
|
|
||||||
}
|
|
||||||
}
|
|
26
build.gradle.kts
Normal file
26
build.gradle.kts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import org.jetbrains.kotlin.gradle.dsl.JvmTarget
|
||||||
|
|
||||||
|
plugins {
|
||||||
|
kotlin("jvm") version "2.1.0"
|
||||||
|
}
|
||||||
|
|
||||||
|
allprojects {
|
||||||
|
apply(plugin="kotlin")
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
mavenLocal()
|
||||||
|
mavenCentral()
|
||||||
|
}
|
||||||
|
|
||||||
|
kotlin {
|
||||||
|
compilerOptions {
|
||||||
|
freeCompilerArgs = listOf("-Xwhen-guards")
|
||||||
|
jvmTarget = JvmTarget.JVM_11
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
java {
|
||||||
|
targetCompatibility = JavaVersion.VERSION_11
|
||||||
|
sourceCompatibility = JavaVersion.VERSION_11
|
||||||
|
}
|
||||||
|
}
|
@ -1,43 +0,0 @@
|
|||||||
|
|
||||||
plugins {
|
|
||||||
id 'java'
|
|
||||||
id 'application'
|
|
||||||
id "org.jetbrains.kotlin.jvm"
|
|
||||||
}
|
|
||||||
|
|
||||||
java {
|
|
||||||
toolchain {
|
|
||||||
languageVersion = JavaLanguageVersion.of(javaVersion)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
compileKotlin {
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = javaVersion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
compileTestKotlin {
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = javaVersion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
dependencies {
|
|
||||||
// should have no dependencies to other modules
|
|
||||||
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk8"
|
|
||||||
implementation "com.michael-bull.kotlin-result:kotlin-result-jvm:1.1.20"
|
|
||||||
}
|
|
||||||
|
|
||||||
sourceSets {
|
|
||||||
main {
|
|
||||||
java {
|
|
||||||
srcDir "${project.projectDir}/src"
|
|
||||||
}
|
|
||||||
resources {
|
|
||||||
srcDir "${project.projectDir}/res"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// note: there are no unit tests in this module!
|
|
24
codeCore/build.gradle.kts
Normal file
24
codeCore/build.gradle.kts
Normal file
@ -0,0 +1,24 @@
|
|||||||
|
import org.jetbrains.kotlin.gradle.dsl.JvmTarget
|
||||||
|
|
||||||
|
plugins {
|
||||||
|
kotlin("jvm")
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
// should have no dependencies to other modules
|
||||||
|
// implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk8")
|
||||||
|
implementation("com.michael-bull.kotlin-result:kotlin-result-jvm:2.0.0")
|
||||||
|
}
|
||||||
|
|
||||||
|
sourceSets {
|
||||||
|
main {
|
||||||
|
java {
|
||||||
|
srcDir("${project.projectDir}/src")
|
||||||
|
}
|
||||||
|
resources {
|
||||||
|
srcDir("${project.projectDir}/res")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// note: there are no unit tests in this module!
|
@ -1,10 +1,14 @@
|
|||||||
package prog8.code
|
package prog8.code
|
||||||
|
|
||||||
|
import prog8.code.ast.PtAsmSub
|
||||||
import prog8.code.ast.PtNode
|
import prog8.code.ast.PtNode
|
||||||
import prog8.code.ast.PtProgram
|
import prog8.code.ast.PtProgram
|
||||||
import prog8.code.core.*
|
import prog8.code.core.*
|
||||||
|
|
||||||
|
|
||||||
|
const val internedStringsModuleName = "prog8_interned_strings"
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tree structure containing all symbol definitions in the program
|
* Tree structure containing all symbol definitions in the program
|
||||||
* (blocks, subroutines, variables (all types), memoryslabs, and labels).
|
* (blocks, subroutines, variables (all types), memoryslabs, and labels).
|
||||||
@ -38,6 +42,7 @@ class SymbolTable(astProgram: PtProgram) : StNode(astProgram.name, StNodeType.GL
|
|||||||
}
|
}
|
||||||
|
|
||||||
val allVariables: Collection<StStaticVariable> by lazy {
|
val allVariables: Collection<StStaticVariable> by lazy {
|
||||||
|
// can't be done with a generic function because those don't support local recursive functions yet
|
||||||
val vars = mutableListOf<StStaticVariable>()
|
val vars = mutableListOf<StStaticVariable>()
|
||||||
fun collect(node: StNode) {
|
fun collect(node: StNode) {
|
||||||
for(child in node.children) {
|
for(child in node.children) {
|
||||||
@ -52,6 +57,7 @@ class SymbolTable(astProgram: PtProgram) : StNode(astProgram.name, StNodeType.GL
|
|||||||
}
|
}
|
||||||
|
|
||||||
val allMemMappedVariables: Collection<StMemVar> by lazy {
|
val allMemMappedVariables: Collection<StMemVar> by lazy {
|
||||||
|
// can't be done with a generic function because those don't support local recursive functions yet
|
||||||
val vars = mutableListOf<StMemVar>()
|
val vars = mutableListOf<StMemVar>()
|
||||||
fun collect(node: StNode) {
|
fun collect(node: StNode) {
|
||||||
for(child in node.children) {
|
for(child in node.children) {
|
||||||
@ -66,6 +72,7 @@ class SymbolTable(astProgram: PtProgram) : StNode(astProgram.name, StNodeType.GL
|
|||||||
}
|
}
|
||||||
|
|
||||||
val allMemorySlabs: Collection<StMemorySlab> by lazy {
|
val allMemorySlabs: Collection<StMemorySlab> by lazy {
|
||||||
|
// can't be done with a generic function because those don't support local recursive functions yet
|
||||||
val vars = mutableListOf<StMemorySlab>()
|
val vars = mutableListOf<StMemorySlab>()
|
||||||
fun collect(node: StNode) {
|
fun collect(node: StNode) {
|
||||||
for(child in node.children) {
|
for(child in node.children) {
|
||||||
@ -82,8 +89,7 @@ class SymbolTable(astProgram: PtProgram) : StNode(astProgram.name, StNodeType.GL
|
|||||||
override fun lookup(scopedName: String) = flat[scopedName]
|
override fun lookup(scopedName: String) = flat[scopedName]
|
||||||
|
|
||||||
fun getLength(name: String): Int? {
|
fun getLength(name: String): Int? {
|
||||||
val node = flat[name]
|
return when(val node = flat[name]) {
|
||||||
return when(node) {
|
|
||||||
is StMemVar -> node.length
|
is StMemVar -> node.length
|
||||||
is StMemorySlab -> node.size.toInt()
|
is StMemorySlab -> node.size.toInt()
|
||||||
is StStaticVariable -> node.length
|
is StStaticVariable -> node.length
|
||||||
@ -98,7 +104,7 @@ enum class StNodeType {
|
|||||||
// MODULE, // not used with current scoping rules
|
// MODULE, // not used with current scoping rules
|
||||||
BLOCK,
|
BLOCK,
|
||||||
SUBROUTINE,
|
SUBROUTINE,
|
||||||
ROMSUB,
|
EXTSUB,
|
||||||
LABEL,
|
LABEL,
|
||||||
STATICVAR,
|
STATICVAR,
|
||||||
MEMVAR,
|
MEMVAR,
|
||||||
@ -110,7 +116,7 @@ enum class StNodeType {
|
|||||||
|
|
||||||
open class StNode(val name: String,
|
open class StNode(val name: String,
|
||||||
val type: StNodeType,
|
val type: StNodeType,
|
||||||
val astNode: PtNode,
|
val astNode: PtNode?,
|
||||||
val children: MutableMap<String, StNode> = mutableMapOf()
|
val children: MutableMap<String, StNode> = mutableMapOf()
|
||||||
) {
|
) {
|
||||||
|
|
||||||
@ -179,37 +185,54 @@ open class StNode(val name: String,
|
|||||||
|
|
||||||
class StStaticVariable(name: String,
|
class StStaticVariable(name: String,
|
||||||
val dt: DataType,
|
val dt: DataType,
|
||||||
val onetimeInitializationNumericValue: Double?, // regular (every-run-time) initialization is done via regular assignments
|
val initializationStringValue: StString?,
|
||||||
val onetimeInitializationStringValue: StString?,
|
val initializationArrayValue: StArray?,
|
||||||
val onetimeInitializationArrayValue: StArray?,
|
|
||||||
val length: Int?, // for arrays: the number of elements, for strings: number of characters *including* the terminating 0-byte
|
val length: Int?, // for arrays: the number of elements, for strings: number of characters *including* the terminating 0-byte
|
||||||
val zpwish: ZeropageWish, // used in the variable allocator
|
val zpwish: ZeropageWish, // used in the variable allocator
|
||||||
astNode: PtNode) : StNode(name, StNodeType.STATICVAR, astNode) {
|
val align: Int,
|
||||||
|
astNode: PtNode?) : StNode(name, StNodeType.STATICVAR, astNode) {
|
||||||
|
|
||||||
val uninitialized = onetimeInitializationArrayValue==null && onetimeInitializationStringValue==null && onetimeInitializationNumericValue==null
|
var initializationNumericValue: Double? = null
|
||||||
|
private set
|
||||||
|
|
||||||
|
fun setOnetimeInitNumeric(number: Double) {
|
||||||
|
// In certain cases the init value of an existing var should be updated,
|
||||||
|
// so we can't ask this as a constructor parameter.
|
||||||
|
// This has to do with the way Prog8 does the (re)initialization of such variables: via code assignment statements.
|
||||||
|
// Certain codegens might want to put them back into the variable directly.
|
||||||
|
// For strings and arrays this doesn't occur - these are always already specced at creation time.
|
||||||
|
initializationNumericValue = number
|
||||||
|
}
|
||||||
|
|
||||||
|
val uninitialized: Boolean
|
||||||
|
get() = initializationArrayValue==null && initializationStringValue==null && initializationNumericValue==null
|
||||||
|
|
||||||
init {
|
init {
|
||||||
if(length!=null) {
|
if(length!=null) {
|
||||||
require(onetimeInitializationNumericValue == null)
|
require(initializationNumericValue == null)
|
||||||
if(onetimeInitializationArrayValue!=null)
|
if(initializationArrayValue!=null)
|
||||||
require(onetimeInitializationArrayValue.isEmpty() ||onetimeInitializationArrayValue.size==length)
|
require(initializationArrayValue.isEmpty() ||initializationArrayValue.size==length)
|
||||||
}
|
}
|
||||||
if(onetimeInitializationNumericValue!=null) {
|
if(initializationNumericValue!=null) {
|
||||||
require(dt in NumericDatatypes || dt==DataType.BOOL)
|
require(dt.isNumericOrBool)
|
||||||
}
|
}
|
||||||
if(onetimeInitializationArrayValue!=null) {
|
if(initializationArrayValue!=null) {
|
||||||
require(dt in ArrayDatatypes)
|
require(dt.isArray)
|
||||||
require(length==onetimeInitializationArrayValue.size)
|
require(length == initializationArrayValue.size)
|
||||||
}
|
}
|
||||||
if(onetimeInitializationStringValue!=null) {
|
if(initializationStringValue!=null) {
|
||||||
require(dt == DataType.STR)
|
require(dt.isString)
|
||||||
require(length == onetimeInitializationStringValue.first.length+1)
|
require(length == initializationStringValue.first.length + 1)
|
||||||
|
}
|
||||||
|
if(align > 0) {
|
||||||
|
require(dt.isString || dt.isArray)
|
||||||
|
require(zpwish != ZeropageWish.REQUIRE_ZEROPAGE && zpwish != ZeropageWish.PREFER_ZEROPAGE)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class StConstant(name: String, val dt: DataType, val value: Double, astNode: PtNode) :
|
class StConstant(name: String, val dt: BaseDataType, val value: Double, astNode: PtNode?) :
|
||||||
StNode(name, StNodeType.CONSTANT, astNode)
|
StNode(name, StNodeType.CONSTANT, astNode)
|
||||||
|
|
||||||
|
|
||||||
@ -217,13 +240,13 @@ class StMemVar(name: String,
|
|||||||
val dt: DataType,
|
val dt: DataType,
|
||||||
val address: UInt,
|
val address: UInt,
|
||||||
val length: Int?, // for arrays: the number of elements, for strings: number of characters *including* the terminating 0-byte
|
val length: Int?, // for arrays: the number of elements, for strings: number of characters *including* the terminating 0-byte
|
||||||
astNode: PtNode) :
|
astNode: PtNode?) :
|
||||||
StNode(name, StNodeType.MEMVAR, astNode) {
|
StNode(name, StNodeType.MEMVAR, astNode) {
|
||||||
|
|
||||||
init{
|
init{
|
||||||
require(dt!=DataType.BOOL && dt!=DataType.ARRAY_BOOL)
|
require(!dt.isString)
|
||||||
if(dt in ArrayDatatypes || dt == DataType.STR)
|
if(dt.isStringly && !dt.isWord)
|
||||||
require(length!=null) { "memory mapped array or string must have known length" }
|
requireNotNull(length)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -231,7 +254,7 @@ class StMemorySlab(
|
|||||||
name: String,
|
name: String,
|
||||||
val size: UInt,
|
val size: UInt,
|
||||||
val align: UInt,
|
val align: UInt,
|
||||||
astNode: PtNode
|
astNode: PtNode?
|
||||||
):
|
):
|
||||||
StNode(name, StNodeType.MEMORYSLAB, astNode)
|
StNode(name, StNodeType.MEMORYSLAB, astNode)
|
||||||
|
|
||||||
@ -240,18 +263,24 @@ class StSub(name: String, val parameters: List<StSubroutineParameter>, val retur
|
|||||||
StNode(name, StNodeType.SUBROUTINE, astNode)
|
StNode(name, StNodeType.SUBROUTINE, astNode)
|
||||||
|
|
||||||
|
|
||||||
class StRomSub(name: String,
|
class StExtSub(name: String,
|
||||||
val address: UInt?, // null in case of asmsub, specified in case of romsub
|
val address: PtAsmSub.Address?, // null in case of asmsub, specified in case of extsub.
|
||||||
val parameters: List<StRomSubParameter>,
|
val parameters: List<StExtSubParameter>,
|
||||||
val returns: List<StRomSubParameter>,
|
val returns: List<StExtSubParameter>,
|
||||||
astNode: PtNode) :
|
astNode: PtNode) :
|
||||||
StNode(name, StNodeType.ROMSUB, astNode)
|
StNode(name, StNodeType.EXTSUB, astNode)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
class StSubroutineParameter(val name: String, val type: DataType)
|
class StSubroutineParameter(val name: String, val type: DataType, val register: RegisterOrPair?)
|
||||||
class StRomSubParameter(val register: RegisterOrStatusflag, val type: DataType)
|
class StExtSubParameter(val register: RegisterOrStatusflag, val type: DataType)
|
||||||
class StArrayElement(val number: Double?, val addressOfSymbol: String?, val boolean: Boolean?)
|
class StArrayElement(val number: Double?, val addressOfSymbol: String?, val boolean: Boolean?) {
|
||||||
|
init {
|
||||||
|
if(number!=null) require(addressOfSymbol==null && boolean==null)
|
||||||
|
if(addressOfSymbol!=null) require(number==null && boolean==null)
|
||||||
|
if(boolean!=null) require(addressOfSymbol==null && number==null)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
typealias StString = Pair<String, Encoding>
|
typealias StString = Pair<String, Encoding>
|
||||||
typealias StArray = List<StArrayElement>
|
typealias StArray = List<StArrayElement>
|
||||||
|
@ -3,18 +3,19 @@ package prog8.code
|
|||||||
import prog8.code.ast.*
|
import prog8.code.ast.*
|
||||||
import prog8.code.core.*
|
import prog8.code.core.*
|
||||||
import prog8.code.target.VMTarget
|
import prog8.code.target.VMTarget
|
||||||
import java.util.*
|
|
||||||
|
|
||||||
class SymbolTableMaker(private val program: PtProgram, private val options: CompilationOptions) {
|
class SymbolTableMaker(private val program: PtProgram, private val options: CompilationOptions) {
|
||||||
fun make(): SymbolTable {
|
fun make(): SymbolTable {
|
||||||
val st = SymbolTable(program)
|
val st = SymbolTable(program)
|
||||||
|
|
||||||
BuiltinFunctions.forEach {
|
BuiltinFunctions.forEach {
|
||||||
st.add(StNode(it.key, StNodeType.BUILTINFUNC, PtIdentifier(it.key, it.value.returnType ?: DataType.UNDEFINED, Position.DUMMY)))
|
val dt = DataType.forDt(it.value.returnType ?: BaseDataType.UNDEFINED)
|
||||||
|
st.add(StNode(it.key, StNodeType.BUILTINFUNC, PtIdentifier(it.key, dt, Position.DUMMY)))
|
||||||
}
|
}
|
||||||
|
|
||||||
val scopestack = Stack<StNode>()
|
val scopestack = ArrayDeque<StNode>()
|
||||||
scopestack.push(st)
|
scopestack.add(st)
|
||||||
program.children.forEach {
|
program.children.forEach {
|
||||||
addToSt(it, scopestack)
|
addToSt(it, scopestack)
|
||||||
}
|
}
|
||||||
@ -22,10 +23,10 @@ class SymbolTableMaker(private val program: PtProgram, private val options: Comp
|
|||||||
|
|
||||||
if(options.compTarget.name != VMTarget.NAME) {
|
if(options.compTarget.name != VMTarget.NAME) {
|
||||||
listOf(
|
listOf(
|
||||||
PtMemMapped("P8ZP_SCRATCH_B1", DataType.UBYTE, options.compTarget.machine.zeropage.SCRATCH_B1, null, Position.DUMMY),
|
PtMemMapped("P8ZP_SCRATCH_B1", DataType.forDt(BaseDataType.UBYTE), options.compTarget.machine.zeropage.SCRATCH_B1, null, Position.DUMMY),
|
||||||
PtMemMapped("P8ZP_SCRATCH_REG", DataType.UBYTE, options.compTarget.machine.zeropage.SCRATCH_REG, null, Position.DUMMY),
|
PtMemMapped("P8ZP_SCRATCH_REG", DataType.forDt(BaseDataType.UBYTE), options.compTarget.machine.zeropage.SCRATCH_REG, null, Position.DUMMY),
|
||||||
PtMemMapped("P8ZP_SCRATCH_W1", DataType.UWORD, options.compTarget.machine.zeropage.SCRATCH_W1, null, Position.DUMMY),
|
PtMemMapped("P8ZP_SCRATCH_W1", DataType.forDt(BaseDataType.UWORD), options.compTarget.machine.zeropage.SCRATCH_W1, null, Position.DUMMY),
|
||||||
PtMemMapped("P8ZP_SCRATCH_W2", DataType.UWORD, options.compTarget.machine.zeropage.SCRATCH_W2, null, Position.DUMMY),
|
PtMemMapped("P8ZP_SCRATCH_W2", DataType.forDt(BaseDataType.UWORD), options.compTarget.machine.zeropage.SCRATCH_W2, null, Position.DUMMY),
|
||||||
).forEach {
|
).forEach {
|
||||||
it.parent = program
|
it.parent = program
|
||||||
st.add(StMemVar(it.name, it.type, it.address, it.arraySize?.toInt(), it))
|
st.add(StMemVar(it.name, it.type, it.address, it.arraySize?.toInt(), it))
|
||||||
@ -35,18 +36,19 @@ class SymbolTableMaker(private val program: PtProgram, private val options: Comp
|
|||||||
return st
|
return st
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun addToSt(node: PtNode, scope: Stack<StNode>) {
|
private fun addToSt(node: PtNode, scope: ArrayDeque<StNode>) {
|
||||||
val stNode = when(node) {
|
val stNode = when(node) {
|
||||||
is PtAsmSub -> {
|
is PtAsmSub -> {
|
||||||
val parameters = node.parameters.map { StRomSubParameter(it.first, it.second.type) }
|
val parameters = node.parameters.map { StExtSubParameter(it.first, it.second.type) }
|
||||||
val returns = node.returns.map { StRomSubParameter(it.first, it.second) }
|
val returns = node.returns.map { StExtSubParameter(it.first, it.second) }
|
||||||
StRomSub(node.name, node.address, parameters, returns, node)
|
StExtSub(node.name, node.address, parameters, returns, node)
|
||||||
}
|
}
|
||||||
is PtBlock -> {
|
is PtBlock -> {
|
||||||
StNode(node.name, StNodeType.BLOCK, node)
|
StNode(node.name, StNodeType.BLOCK, node)
|
||||||
}
|
}
|
||||||
is PtConstant -> {
|
is PtConstant -> {
|
||||||
StConstant(node.name, node.type, node.value, node)
|
require(node.type.isNumericOrBool)
|
||||||
|
StConstant(node.name, node.type.base, node.value, node)
|
||||||
}
|
}
|
||||||
is PtLabel -> {
|
is PtLabel -> {
|
||||||
StNode(node.name, StNodeType.LABEL, node)
|
StNode(node.name, StNodeType.LABEL, node)
|
||||||
@ -55,7 +57,7 @@ class SymbolTableMaker(private val program: PtProgram, private val options: Comp
|
|||||||
StMemVar(node.name, node.type, node.address, node.arraySize?.toInt(), node)
|
StMemVar(node.name, node.type, node.address, node.arraySize?.toInt(), node)
|
||||||
}
|
}
|
||||||
is PtSub -> {
|
is PtSub -> {
|
||||||
val params = node.parameters.map {StSubroutineParameter(it.name, it.type) }
|
val params = node.parameters.map {StSubroutineParameter(it.name, it.type, it.register) }
|
||||||
StSub(node.name, params, node.returntype, node)
|
StSub(node.name, params, node.returntype, node)
|
||||||
}
|
}
|
||||||
is PtVariable -> {
|
is PtVariable -> {
|
||||||
@ -65,23 +67,26 @@ class SymbolTableMaker(private val program: PtProgram, private val options: Comp
|
|||||||
val numElements: Int?
|
val numElements: Int?
|
||||||
val value = node.value
|
val value = node.value
|
||||||
if(value!=null) {
|
if(value!=null) {
|
||||||
val number = (value as? PtNumber)?.number
|
|
||||||
initialNumeric = if(number==0.0) null else number // 0 as init value -> just uninitialized
|
|
||||||
when (value) {
|
when (value) {
|
||||||
is PtString -> {
|
is PtString -> {
|
||||||
initialString = StString(value.value, value.encoding)
|
initialString = StString(value.value, value.encoding)
|
||||||
initialArray = null
|
initialArray = null
|
||||||
|
initialNumeric = null
|
||||||
numElements = value.value.length + 1 // include the terminating 0-byte
|
numElements = value.value.length + 1 // include the terminating 0-byte
|
||||||
}
|
}
|
||||||
is PtArray -> {
|
is PtArray -> {
|
||||||
initialArray = makeInitialArray(value)
|
initialArray = makeInitialArray(value)
|
||||||
initialString = null
|
initialString = null
|
||||||
|
initialNumeric = null
|
||||||
numElements = initialArray.size
|
numElements = initialArray.size
|
||||||
require(node.arraySize?.toInt()==numElements)
|
require(node.arraySize?.toInt()==numElements)
|
||||||
}
|
}
|
||||||
else -> {
|
else -> {
|
||||||
|
require(value is PtNumber)
|
||||||
initialString = null
|
initialString = null
|
||||||
initialArray = null
|
initialArray = null
|
||||||
|
val number = value.number
|
||||||
|
initialNumeric = number
|
||||||
numElements = node.arraySize?.toInt()
|
numElements = node.arraySize?.toInt()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -94,7 +99,10 @@ class SymbolTableMaker(private val program: PtProgram, private val options: Comp
|
|||||||
// if(node.type in SplitWordArrayTypes) {
|
// if(node.type in SplitWordArrayTypes) {
|
||||||
// ... split array also add _lsb and _msb to symboltable?
|
// ... split array also add _lsb and _msb to symboltable?
|
||||||
// }
|
// }
|
||||||
StStaticVariable(node.name, node.type, initialNumeric, initialString, initialArray, numElements, node.zeropage, node)
|
val stVar = StStaticVariable(node.name, node.type, initialString, initialArray, numElements, node.zeropage, node.align.toInt(), node)
|
||||||
|
if(initialNumeric!=null)
|
||||||
|
stVar.setOnetimeInitNumeric(initialNumeric)
|
||||||
|
stVar
|
||||||
}
|
}
|
||||||
is PtBuiltinFunctionCall -> {
|
is PtBuiltinFunctionCall -> {
|
||||||
if(node.name=="memory") {
|
if(node.name=="memory") {
|
||||||
@ -104,7 +112,7 @@ class SymbolTableMaker(private val program: PtProgram, private val options: Comp
|
|||||||
val size = (node.args[1] as PtNumber).number.toUInt()
|
val size = (node.args[1] as PtNumber).number.toUInt()
|
||||||
val align = (node.args[2] as PtNumber).number.toUInt()
|
val align = (node.args[2] as PtNumber).number.toUInt()
|
||||||
// don't add memory slabs in nested scope, just put them in the top level of the ST
|
// don't add memory slabs in nested scope, just put them in the top level of the ST
|
||||||
scope.firstElement().add(StMemorySlab("prog8_memoryslab_$slabname", size, align, node))
|
scope.first().add(StMemorySlab("prog8_memoryslab_$slabname", size, align, node))
|
||||||
}
|
}
|
||||||
null
|
null
|
||||||
}
|
}
|
||||||
@ -112,14 +120,14 @@ class SymbolTableMaker(private val program: PtProgram, private val options: Comp
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(stNode!=null) {
|
if(stNode!=null) {
|
||||||
scope.peek().add(stNode)
|
scope.last().add(stNode)
|
||||||
scope.push(stNode)
|
scope.add(stNode)
|
||||||
}
|
}
|
||||||
node.children.forEach {
|
node.children.forEach {
|
||||||
addToSt(it, scope)
|
addToSt(it, scope)
|
||||||
}
|
}
|
||||||
if(stNode!=null)
|
if(stNode!=null)
|
||||||
scope.pop()
|
scope.removeLast()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun makeInitialArray(value: PtArray): List<StArrayElement> {
|
private fun makeInitialArray(value: PtArray): List<StArrayElement> {
|
||||||
@ -130,7 +138,6 @@ class SymbolTableMaker(private val program: PtProgram, private val options: Comp
|
|||||||
TODO("address-of array element $it in initial array value")
|
TODO("address-of array element $it in initial array value")
|
||||||
StArrayElement(null, it.identifier.name, null)
|
StArrayElement(null, it.identifier.name, null)
|
||||||
}
|
}
|
||||||
is PtIdentifier -> StArrayElement(null, it.name, null)
|
|
||||||
is PtNumber -> StArrayElement(it.number, null, null)
|
is PtNumber -> StArrayElement(it.number, null, null)
|
||||||
is PtBool -> StArrayElement(null, null, it.value)
|
is PtBool -> StArrayElement(null, null, it.value)
|
||||||
else -> throw AssemblyError("invalid array element $it")
|
else -> throw AssemblyError("invalid array element $it")
|
||||||
|
@ -3,7 +3,7 @@ package prog8.code.ast
|
|||||||
import prog8.code.core.IMemSizer
|
import prog8.code.core.IMemSizer
|
||||||
import prog8.code.core.IStringEncoding
|
import prog8.code.core.IStringEncoding
|
||||||
import prog8.code.core.Position
|
import prog8.code.core.Position
|
||||||
import prog8.code.core.SourceCode
|
import prog8.code.source.SourceCode
|
||||||
import java.nio.file.Path
|
import java.nio.file.Path
|
||||||
|
|
||||||
// New simplified AST for the code generator.
|
// New simplified AST for the code generator.
|
||||||
@ -80,18 +80,11 @@ class PtBlock(name: String,
|
|||||||
val options: Options,
|
val options: Options,
|
||||||
position: Position
|
position: Position
|
||||||
) : PtNamedNode(name, position), IPtStatementContainer {
|
) : PtNamedNode(name, position), IPtStatementContainer {
|
||||||
enum class BlockAlignment {
|
|
||||||
NONE,
|
|
||||||
WORD,
|
|
||||||
PAGE
|
|
||||||
}
|
|
||||||
|
|
||||||
class Options(val address: UInt? = null,
|
class Options(val address: UInt? = null,
|
||||||
val forceOutput: Boolean = false,
|
val forceOutput: Boolean = false,
|
||||||
val noSymbolPrefixing: Boolean = false,
|
val noSymbolPrefixing: Boolean = false,
|
||||||
val veraFxMuls: Boolean = false,
|
val veraFxMuls: Boolean = false,
|
||||||
val ignoreUnused: Boolean = false,
|
val ignoreUnused: Boolean = false)
|
||||||
val alignment: BlockAlignment = BlockAlignment.NONE)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -103,12 +96,20 @@ class PtInlineAssembly(val assembly: String, val isIR: Boolean, position: Positi
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtLabel(name: String, position: Position) : PtNamedNode(name, position)
|
class PtLabel(name: String, position: Position) : PtNamedNode(name, position) {
|
||||||
|
companion object {
|
||||||
|
// all automatically generated labels everywhere need to have the same label name prefix:
|
||||||
|
const val GENERATED_LABEL_PREFIX = "p8_label_gen_"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtBreakpoint(position: Position): PtNode(position)
|
class PtBreakpoint(position: Position): PtNode(position)
|
||||||
|
|
||||||
|
|
||||||
|
class PtAlign(val align: UInt, position: Position): PtNode(position)
|
||||||
|
|
||||||
|
|
||||||
class PtIncludeBinary(val file: Path, val offset: UInt?, val length: UInt?, position: Position) : PtNode(position)
|
class PtIncludeBinary(val file: Path, val offset: UInt?, val length: UInt?, position: Position) : PtNode(position)
|
||||||
|
|
||||||
|
|
||||||
|
@ -9,7 +9,7 @@ import kotlin.math.truncate
|
|||||||
sealed class PtExpression(val type: DataType, position: Position) : PtNode(position) {
|
sealed class PtExpression(val type: DataType, position: Position) : PtNode(position) {
|
||||||
|
|
||||||
init {
|
init {
|
||||||
if(type==DataType.UNDEFINED) {
|
if(type.isUndefined) {
|
||||||
@Suppress("LeakingThis")
|
@Suppress("LeakingThis")
|
||||||
when(this) {
|
when(this) {
|
||||||
is PtBuiltinFunctionCall -> { /* void function call */ }
|
is PtBuiltinFunctionCall -> { /* void function call */ }
|
||||||
@ -42,9 +42,20 @@ sealed class PtExpression(val type: DataType, position: Position) : PtNode(posit
|
|||||||
else
|
else
|
||||||
other.left isSameAs left && other.right isSameAs right
|
other.left isSameAs left && other.right isSameAs right
|
||||||
}
|
}
|
||||||
is PtContainmentCheck -> other is PtContainmentCheck && other.type==type && other.element isSameAs element && other.iterable isSameAs iterable
|
is PtContainmentCheck -> {
|
||||||
|
if(other !is PtContainmentCheck || other.type != type || !(other.needle isSameAs needle))
|
||||||
|
false
|
||||||
|
else {
|
||||||
|
if(haystackHeapVar!=null)
|
||||||
|
other.haystackHeapVar!=null && other.haystackHeapVar!! isSameAs haystackHeapVar!!
|
||||||
|
else if(haystackValues!=null)
|
||||||
|
other.haystackValues!=null && other.haystackValues!! isSameAs haystackValues!!
|
||||||
|
else
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}
|
||||||
is PtIdentifier -> other is PtIdentifier && other.type==type && other.name==name
|
is PtIdentifier -> other is PtIdentifier && other.type==type && other.name==name
|
||||||
is PtMachineRegister -> other is PtMachineRegister && other.type==type && other.register==register
|
is PtIrRegister -> other is PtIrRegister && other.type==type && other.register==register
|
||||||
is PtMemoryByte -> other is PtMemoryByte && other.address isSameAs address
|
is PtMemoryByte -> other is PtMemoryByte && other.address isSameAs address
|
||||||
is PtNumber -> other is PtNumber && other.type==type && other.number==number
|
is PtNumber -> other is PtNumber && other.type==type && other.number==number
|
||||||
is PtBool -> other is PtBool && other.value==value
|
is PtBool -> other is PtBool && other.value==value
|
||||||
@ -55,39 +66,38 @@ sealed class PtExpression(val type: DataType, position: Position) : PtNode(posit
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
infix fun isSameAs(target: PtAssignTarget): Boolean {
|
infix fun isSameAs(target: PtAssignTarget): Boolean = when {
|
||||||
return when {
|
target.memory != null && this is PtMemoryByte -> {
|
||||||
target.memory != null && this is PtMemoryByte-> {
|
target.memory!!.address isSameAs this.address
|
||||||
target.memory!!.address isSameAs this.address
|
|
||||||
}
|
|
||||||
target.identifier != null && this is PtIdentifier -> {
|
|
||||||
this.name == target.identifier!!.name
|
|
||||||
}
|
|
||||||
target.array != null && this is PtArrayIndexer -> {
|
|
||||||
this.variable.name == target.array!!.variable.name && this.index isSameAs target.array!!.index && this.splitWords==target.array!!.splitWords
|
|
||||||
}
|
|
||||||
else -> false
|
|
||||||
}
|
}
|
||||||
|
target.identifier != null && this is PtIdentifier -> {
|
||||||
|
this.name == target.identifier!!.name
|
||||||
|
}
|
||||||
|
target.array != null && this is PtArrayIndexer -> {
|
||||||
|
this.variable.name == target.array!!.variable.name && this.index isSameAs target.array!!.index && this.splitWords==target.array!!.splitWords
|
||||||
|
}
|
||||||
|
else -> false
|
||||||
}
|
}
|
||||||
|
|
||||||
fun asConstInteger(): Int? = (this as? PtNumber)?.number?.toInt() ?: (this as? PtBool)?.asInt()
|
fun asConstInteger(): Int? = (this as? PtNumber)?.number?.toInt() ?: (this as? PtBool)?.asInt()
|
||||||
|
fun asConstValue(): Double? = (this as? PtNumber)?.number ?: (this as? PtBool)?.asInt()?.toDouble()
|
||||||
|
|
||||||
fun isSimple(): Boolean {
|
fun isSimple(): Boolean {
|
||||||
return when(this) {
|
return when(this) {
|
||||||
is PtAddressOf -> true
|
is PtAddressOf -> this.arrayIndexExpr==null || this.arrayIndexExpr?.isSimple()==true
|
||||||
is PtArray -> true
|
is PtArray -> true
|
||||||
is PtArrayIndexer -> index is PtNumber || index is PtIdentifier
|
is PtArrayIndexer -> index is PtNumber || index is PtIdentifier
|
||||||
is PtBinaryExpression -> false
|
is PtBinaryExpression -> false
|
||||||
is PtBuiltinFunctionCall -> {
|
is PtBuiltinFunctionCall -> {
|
||||||
when (name) {
|
when (name) {
|
||||||
in arrayOf("msb", "lsb", "mkword", "set_carry", "set_irqd", "clear_carry", "clear_irqd") -> this.args.all { it.isSimple() }
|
in arrayOf("msb", "lsb", "msw", "lsw", "mkword", "set_carry", "set_irqd", "clear_carry", "clear_irqd") -> this.args.all { it.isSimple() }
|
||||||
else -> false
|
else -> false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
is PtContainmentCheck -> false
|
is PtContainmentCheck -> false
|
||||||
is PtFunctionCall -> false
|
is PtFunctionCall -> false
|
||||||
is PtIdentifier -> true
|
is PtIdentifier -> true
|
||||||
is PtMachineRegister -> true
|
is PtIrRegister -> true
|
||||||
is PtMemoryByte -> address is PtNumber || address is PtIdentifier
|
is PtMemoryByte -> address is PtNumber || address is PtIdentifier
|
||||||
is PtBool -> true
|
is PtBool -> true
|
||||||
is PtNumber -> true
|
is PtNumber -> true
|
||||||
@ -95,6 +105,7 @@ sealed class PtExpression(val type: DataType, position: Position) : PtNode(posit
|
|||||||
is PtRange -> true
|
is PtRange -> true
|
||||||
is PtString -> true
|
is PtString -> true
|
||||||
is PtTypeCast -> value.isSimple()
|
is PtTypeCast -> value.isSimple()
|
||||||
|
is PtIfExpression -> condition.isSimple() && truevalue.isSimple() && falsevalue.isSimple()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -126,7 +137,7 @@ sealed class PtExpression(val type: DataType, position: Position) : PtNode(posit
|
|||||||
*/
|
*/
|
||||||
}
|
}
|
||||||
|
|
||||||
class PtAddressOf(position: Position) : PtExpression(DataType.UWORD, position) {
|
class PtAddressOf(position: Position, val isMsbForSplitArray: Boolean=false) : PtExpression(DataType.forDt(BaseDataType.UWORD), position) {
|
||||||
val identifier: PtIdentifier
|
val identifier: PtIdentifier
|
||||||
get() = children[0] as PtIdentifier
|
get() = children[0] as PtIdentifier
|
||||||
val arrayIndexExpr: PtExpression?
|
val arrayIndexExpr: PtExpression?
|
||||||
@ -139,23 +150,20 @@ class PtAddressOf(position: Position) : PtExpression(DataType.UWORD, position) {
|
|||||||
|
|
||||||
class PtArrayIndexer(elementType: DataType, position: Position): PtExpression(elementType, position) {
|
class PtArrayIndexer(elementType: DataType, position: Position): PtExpression(elementType, position) {
|
||||||
val variable: PtIdentifier
|
val variable: PtIdentifier
|
||||||
get() {
|
get() = children[0] as PtIdentifier
|
||||||
require((children[0] as? PtIdentifier)?.type in ArrayDatatypes+DataType.STR) // TODO remove
|
|
||||||
return children[0] as PtIdentifier
|
|
||||||
}
|
|
||||||
val index: PtExpression
|
val index: PtExpression
|
||||||
get() = children[1] as PtExpression
|
get() = children[1] as PtExpression
|
||||||
|
|
||||||
val splitWords: Boolean
|
val splitWords: Boolean
|
||||||
get() = variable.type in SplitWordArrayTypes
|
get() = variable.type.isSplitWordArray
|
||||||
|
|
||||||
init {
|
init {
|
||||||
require(elementType in NumericDatatypesWithBoolean)
|
require(elementType.isNumericOrBool)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtArray(type: DataType, position: Position): PtExpression(type, position) {
|
class PtArray(type: DataType, position: Position): PtExpression(type, position) {
|
||||||
|
// children are always one of 3 types: PtBool, PtNumber or PtAddressOf.
|
||||||
override fun hashCode(): Int = Objects.hash(children, type)
|
override fun hashCode(): Int = Objects.hash(children, type)
|
||||||
override fun equals(other: Any?): Boolean {
|
override fun equals(other: Any?): Boolean {
|
||||||
if(other==null || other !is PtArray)
|
if(other==null || other !is PtArray)
|
||||||
@ -175,7 +183,7 @@ class PtBuiltinFunctionCall(val name: String,
|
|||||||
position: Position) : PtExpression(type, position) {
|
position: Position) : PtExpression(type, position) {
|
||||||
init {
|
init {
|
||||||
if(!void)
|
if(!void)
|
||||||
require(type!=DataType.UNDEFINED)
|
require(!type.isUndefined)
|
||||||
}
|
}
|
||||||
|
|
||||||
val args: List<PtExpression>
|
val args: List<PtExpression>
|
||||||
@ -191,18 +199,35 @@ class PtBinaryExpression(val operator: String, type: DataType, position: Positio
|
|||||||
|
|
||||||
init {
|
init {
|
||||||
if(operator in ComparisonOperators + LogicalOperators)
|
if(operator in ComparisonOperators + LogicalOperators)
|
||||||
require(type==DataType.BOOL)
|
require(type.isBool)
|
||||||
else
|
else
|
||||||
require(type!=DataType.BOOL) { "no bool allowed for this operator $operator"}
|
require(!type.isBool) { "no bool allowed for this operator $operator"}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtContainmentCheck(position: Position): PtExpression(DataType.BOOL, position) {
|
class PtIfExpression(type: DataType, position: Position): PtExpression(type, position) {
|
||||||
val element: PtExpression
|
val condition: PtExpression
|
||||||
get() = children[0] as PtExpression
|
get() = children[0] as PtExpression
|
||||||
val iterable: PtIdentifier
|
val truevalue: PtExpression
|
||||||
get() = children[1] as PtIdentifier
|
get() = children[1] as PtExpression
|
||||||
|
val falsevalue: PtExpression
|
||||||
|
get() = children[2] as PtExpression
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class PtContainmentCheck(position: Position): PtExpression(DataType.forDt(BaseDataType.BOOL), position) {
|
||||||
|
val needle: PtExpression
|
||||||
|
get() = children[0] as PtExpression
|
||||||
|
val haystackHeapVar: PtIdentifier?
|
||||||
|
get() = children[1] as? PtIdentifier
|
||||||
|
val haystackValues: PtArray?
|
||||||
|
get() = children[1] as? PtArray
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
const val MAX_SIZE_FOR_INLINE_CHECKS_BYTE = 5
|
||||||
|
const val MAX_SIZE_FOR_INLINE_CHECKS_WORD = 4
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -210,13 +235,15 @@ class PtFunctionCall(val name: String,
|
|||||||
val void: Boolean,
|
val void: Boolean,
|
||||||
type: DataType,
|
type: DataType,
|
||||||
position: Position) : PtExpression(type, position) {
|
position: Position) : PtExpression(type, position) {
|
||||||
init {
|
|
||||||
if(!void)
|
|
||||||
require(type!=DataType.UNDEFINED)
|
|
||||||
}
|
|
||||||
|
|
||||||
val args: List<PtExpression>
|
val args: List<PtExpression>
|
||||||
get() = children.map { it as PtExpression }
|
get() = children.map { it as PtExpression }
|
||||||
|
|
||||||
|
init {
|
||||||
|
if(void) require(type.isUndefined) {
|
||||||
|
"void fcall should have undefined datatype"
|
||||||
|
}
|
||||||
|
// note: non-void calls can have UNDEFINED type: is if they return more than 1 value
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -229,19 +256,13 @@ class PtIdentifier(val name: String, type: DataType, position: Position) : PtExp
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtMemoryByte(position: Position) : PtExpression(DataType.UBYTE, position) {
|
class PtMemoryByte(position: Position) : PtExpression(DataType.forDt(BaseDataType.UBYTE), position) {
|
||||||
val address: PtExpression
|
val address: PtExpression
|
||||||
get() = children.single() as PtExpression
|
get() = children.single() as PtExpression
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtBool(val value: Boolean, position: Position) : PtExpression(DataType.BOOL, position) {
|
class PtBool(val value: Boolean, position: Position) : PtExpression(DataType.forDt(BaseDataType.BOOL), position) {
|
||||||
|
|
||||||
companion object {
|
|
||||||
fun fromNumber(number: Number, position: Position): PtBool =
|
|
||||||
PtBool(if(number==0.0) false else true, position)
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun hashCode(): Int = Objects.hash(type, value)
|
override fun hashCode(): Int = Objects.hash(type, value)
|
||||||
|
|
||||||
override fun equals(other: Any?): Boolean {
|
override fun equals(other: Any?): Boolean {
|
||||||
@ -256,32 +277,40 @@ class PtBool(val value: Boolean, position: Position) : PtExpression(DataType.BOO
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtNumber(type: DataType, val number: Double, position: Position) : PtExpression(type, position) {
|
class PtNumber(type: BaseDataType, val number: Double, position: Position) : PtExpression(DataType.forDt(type), position) {
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
fun fromBoolean(bool: Boolean, position: Position): PtNumber =
|
fun fromBoolean(bool: Boolean, position: Position): PtNumber =
|
||||||
PtNumber(DataType.UBYTE, if(bool) 1.0 else 0.0, position)
|
PtNumber(BaseDataType.UBYTE, if(bool) 1.0 else 0.0, position)
|
||||||
}
|
}
|
||||||
|
|
||||||
init {
|
init {
|
||||||
if(type==DataType.BOOL)
|
if(type==BaseDataType.BOOL)
|
||||||
throw IllegalArgumentException("use PtBool instead")
|
throw IllegalArgumentException("use PtBool instead")
|
||||||
if(type!=DataType.FLOAT) {
|
if(type!=BaseDataType.FLOAT) {
|
||||||
val trunc = truncate(number)
|
val trunc = truncate(number)
|
||||||
if (trunc != number)
|
if (trunc != number)
|
||||||
throw IllegalArgumentException("refused truncating of float to avoid loss of precision @$position")
|
throw IllegalArgumentException("refused truncating of float to avoid loss of precision @$position")
|
||||||
}
|
}
|
||||||
|
when(type) {
|
||||||
|
BaseDataType.UBYTE -> require(number in 0.0..255.0)
|
||||||
|
BaseDataType.BYTE -> require(number in -128.0..127.0)
|
||||||
|
BaseDataType.UWORD -> require(number in 0.0..65535.0)
|
||||||
|
BaseDataType.WORD -> require(number in -32728.0..32767.0)
|
||||||
|
BaseDataType.LONG -> require(number in -2147483647.0..2147483647.0)
|
||||||
|
else -> require(type.isNumeric) { "numeric literal type should be numeric: $type" }
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun hashCode(): Int = Objects.hash(type, number)
|
override fun hashCode(): Int = Objects.hash(type, number)
|
||||||
|
|
||||||
override fun equals(other: Any?): Boolean {
|
override fun equals(other: Any?): Boolean {
|
||||||
if(other==null || other !is PtNumber)
|
return if(other==null || other !is PtNumber)
|
||||||
return false
|
false
|
||||||
else if(type!=DataType.BOOL && other.type!=DataType.BOOL)
|
else if(!type.isBool && !other.type.isBool)
|
||||||
return number==other.number
|
number==other.number
|
||||||
else
|
else
|
||||||
return type==other.type && number==other.number
|
type==other.type && number==other.number
|
||||||
}
|
}
|
||||||
|
|
||||||
operator fun compareTo(other: PtNumber): Int = number.compareTo(other.number)
|
operator fun compareTo(other: PtNumber): Int = number.compareTo(other.number)
|
||||||
@ -295,7 +324,7 @@ class PtPrefix(val operator: String, type: DataType, position: Position): PtExpr
|
|||||||
get() = children.single() as PtExpression
|
get() = children.single() as PtExpression
|
||||||
|
|
||||||
init {
|
init {
|
||||||
require(operator in setOf("+", "-", "~", "not")) { "invalid prefix operator: $operator" }
|
require(operator in PrefixOperators) { "invalid prefix operator: $operator" }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -311,6 +340,7 @@ class PtRange(type: DataType, position: Position) : PtExpression(type, position)
|
|||||||
fun toConstantIntegerRange(): IntProgression? {
|
fun toConstantIntegerRange(): IntProgression? {
|
||||||
fun makeRange(fromVal: Int, toVal: Int, stepVal: Int): IntProgression {
|
fun makeRange(fromVal: Int, toVal: Int, stepVal: Int): IntProgression {
|
||||||
return when {
|
return when {
|
||||||
|
fromVal == toVal -> fromVal .. toVal
|
||||||
fromVal <= toVal -> when {
|
fromVal <= toVal -> when {
|
||||||
stepVal <= 0 -> IntRange.EMPTY
|
stepVal <= 0 -> IntRange.EMPTY
|
||||||
stepVal == 1 -> fromVal..toVal
|
stepVal == 1 -> fromVal..toVal
|
||||||
@ -326,18 +356,17 @@ class PtRange(type: DataType, position: Position) : PtExpression(type, position)
|
|||||||
|
|
||||||
val fromLv = from as? PtNumber
|
val fromLv = from as? PtNumber
|
||||||
val toLv = to as? PtNumber
|
val toLv = to as? PtNumber
|
||||||
val stepLv = step as? PtNumber
|
if(fromLv==null || toLv==null)
|
||||||
if(fromLv==null || toLv==null || stepLv==null)
|
|
||||||
return null
|
return null
|
||||||
val fromVal = fromLv.number.toInt()
|
val fromVal = fromLv.number.toInt()
|
||||||
val toVal = toLv.number.toInt()
|
val toVal = toLv.number.toInt()
|
||||||
val stepVal = stepLv.number.toInt()
|
val stepVal = step.number.toInt()
|
||||||
return makeRange(fromVal, toVal, stepVal)
|
return makeRange(fromVal, toVal, stepVal)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtString(val value: String, val encoding: Encoding, position: Position) : PtExpression(DataType.STR, position) {
|
class PtString(val value: String, val encoding: Encoding, position: Position) : PtExpression(DataType.forDt(BaseDataType.STR), position) {
|
||||||
override fun hashCode(): Int = Objects.hash(value, encoding)
|
override fun hashCode(): Int = Objects.hash(value, encoding)
|
||||||
override fun equals(other: Any?): Boolean {
|
override fun equals(other: Any?): Boolean {
|
||||||
if(other==null || other !is PtString)
|
if(other==null || other !is PtString)
|
||||||
@ -347,15 +376,11 @@ class PtString(val value: String, val encoding: Encoding, position: Position) :
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtTypeCast(type: DataType, position: Position) : PtExpression(type, position) {
|
class PtTypeCast(type: BaseDataType, position: Position) : PtExpression(DataType.forDt(type), position) {
|
||||||
val value: PtExpression
|
val value: PtExpression
|
||||||
get() = children.single() as PtExpression
|
get() = children.single() as PtExpression
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// special node that isn't created from compiling user code, but used internally in the Intermediate Code
|
// special node that isn't created from compiling user code, but used internally in the Intermediate Code
|
||||||
class PtMachineRegister(val register: Int, type: DataType, position: Position) : PtExpression(type, position)
|
class PtIrRegister(val register: Int, type: DataType, position: Position) : PtExpression(type, position)
|
||||||
|
|
||||||
|
|
||||||
fun constValue(expr: PtExpression): Double? = if(expr is PtNumber) expr.number else if(expr is PtBool) expr.asInt().toDouble() else null
|
|
||||||
fun constIntValue(expr: PtExpression): Int? = if(expr is PtNumber) expr.number.toInt() else if(expr is PtBool) expr.asInt() else null
|
|
||||||
|
@ -1,27 +1,43 @@
|
|||||||
package prog8.code.ast
|
package prog8.code.ast
|
||||||
|
|
||||||
import prog8.code.core.*
|
import prog8.code.core.DataType
|
||||||
|
import prog8.code.core.escape
|
||||||
|
import prog8.code.core.toHex
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Produces readable text from a [PtNode] (AST node, usually starting with PtProgram as root),
|
* Produces readable text from a [PtNode] (AST node, usually starting with PtProgram as root),
|
||||||
* passing it as a String to the specified receiver function.
|
* passing it as a String to the specified receiver function.
|
||||||
*/
|
*/
|
||||||
fun printAst(root: PtNode, skipLibraries: Boolean, output: (text: String) -> Unit) {
|
fun printAst(root: PtNode, skipLibraries: Boolean, output: (text: String) -> Unit) {
|
||||||
fun type(dt: DataType) = "!${dt.name.lowercase()}!"
|
fun type(dt: DataType) = "!${dt}!"
|
||||||
fun txt(node: PtNode): String {
|
fun txt(node: PtNode): String {
|
||||||
return when(node) {
|
return when(node) {
|
||||||
is PtAssignTarget -> "<target>"
|
is PtAlign -> "%align ${node.align}"
|
||||||
|
is PtAssignTarget -> if(node.void) "<void>" else "<target>"
|
||||||
is PtAssignment -> "<assign>"
|
is PtAssignment -> "<assign>"
|
||||||
is PtAugmentedAssign -> "<inplace-assign> ${node.operator}"
|
is PtAugmentedAssign -> "<inplace-assign> ${node.operator}"
|
||||||
is PtBreakpoint -> "%breakpoint"
|
is PtBreakpoint -> "%breakpoint"
|
||||||
is PtConditionalBranch -> "if_${node.condition.name.lowercase()}"
|
is PtConditionalBranch -> "if_${node.condition.name.lowercase()}"
|
||||||
is PtAddressOf -> {
|
is PtAddressOf -> {
|
||||||
if(node.isFromArrayElement)
|
if(node.isMsbForSplitArray)
|
||||||
|
"&>"
|
||||||
|
else if(node.isFromArrayElement)
|
||||||
"& array-element"
|
"& array-element"
|
||||||
else
|
else
|
||||||
"&"
|
"&"
|
||||||
}
|
}
|
||||||
is PtArray -> "array len=${node.children.size} ${type(node.type)}"
|
is PtArray -> {
|
||||||
|
val valuelist = node.children.joinToString(", ") {
|
||||||
|
when (it) {
|
||||||
|
is PtBool -> it.toString()
|
||||||
|
is PtNumber -> it.number.toString()
|
||||||
|
is PtIdentifier -> it.name
|
||||||
|
is PtAddressOf -> "& ${it.identifier.name}"
|
||||||
|
else -> "invalid array element $it"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"array len=${node.children.size} ${type(node.type)} [ $valuelist ]"
|
||||||
|
}
|
||||||
is PtArrayIndexer -> "<arrayindexer> ${type(node.type)} ${if(node.splitWords) "[splitwords]" else ""}"
|
is PtArrayIndexer -> "<arrayindexer> ${type(node.type)} ${if(node.splitWords) "[splitwords]" else ""}"
|
||||||
is PtBinaryExpression -> "<expr> ${node.operator} ${type(node.type)}"
|
is PtBinaryExpression -> "<expr> ${node.operator} ${type(node.type)}"
|
||||||
is PtBuiltinFunctionCall -> {
|
is PtBuiltinFunctionCall -> {
|
||||||
@ -34,17 +50,17 @@ fun printAst(root: PtNode, skipLibraries: Boolean, output: (text: String) -> Uni
|
|||||||
str + node.name + "()"
|
str + node.name + "()"
|
||||||
}
|
}
|
||||||
is PtIdentifier -> "${node.name} ${type(node.type)}"
|
is PtIdentifier -> "${node.name} ${type(node.type)}"
|
||||||
is PtMachineRegister -> "VMREG#${node.register} ${type(node.type)}"
|
is PtIrRegister -> "IRREG#${node.register} ${type(node.type)}"
|
||||||
is PtMemoryByte -> "@()"
|
is PtMemoryByte -> "@()"
|
||||||
is PtNumber -> {
|
is PtNumber -> {
|
||||||
val numstr = if(node.type == DataType.FLOAT) node.number.toString() else node.number.toHex()
|
val numstr = if(node.type.isFloat) node.number.toString() else node.number.toHex()
|
||||||
"$numstr ${type(node.type)}"
|
"$numstr ${type(node.type)}"
|
||||||
}
|
}
|
||||||
is PtBool -> node.value.toString()
|
is PtBool -> node.value.toString()
|
||||||
is PtPrefix -> node.operator
|
is PtPrefix -> node.operator
|
||||||
is PtRange -> "<range>"
|
is PtRange -> "<range>"
|
||||||
is PtString -> "\"${node.value.escape()}\""
|
is PtString -> "\"${node.value.escape()}\""
|
||||||
is PtTypeCast -> "as ${node.type.name.lowercase()}"
|
is PtTypeCast -> "as ${node.type}"
|
||||||
is PtForLoop -> "for"
|
is PtForLoop -> "for"
|
||||||
is PtIfElse -> "ifelse"
|
is PtIfElse -> "ifelse"
|
||||||
is PtIncludeBinary -> "%incbin '${node.file}', ${node.offset}, ${node.length}"
|
is PtIncludeBinary -> "%incbin '${node.file}', ${node.offset}, ${node.length}"
|
||||||
@ -55,77 +71,83 @@ fun printAst(root: PtNode, skipLibraries: Boolean, output: (text: String) -> Uni
|
|||||||
"%asm {{ ...${node.assembly.length} characters... }}"
|
"%asm {{ ...${node.assembly.length} characters... }}"
|
||||||
}
|
}
|
||||||
is PtJump -> {
|
is PtJump -> {
|
||||||
if(node.identifier!=null)
|
"goto ${txt(node.target)}"
|
||||||
"goto ${node.identifier.name}"
|
|
||||||
else if(node.address!=null)
|
|
||||||
"goto ${node.address.toHex()}"
|
|
||||||
else
|
|
||||||
"???"
|
|
||||||
}
|
}
|
||||||
is PtAsmSub -> {
|
is PtAsmSub -> {
|
||||||
val params = node.parameters.map {
|
val params = node.parameters.joinToString(", ") {
|
||||||
val register = it.first.registerOrPair
|
val register = it.first.registerOrPair
|
||||||
val statusflag = it.first.statusflag
|
val statusflag = it.first.statusflag
|
||||||
"${it.second.type} ${it.second.name} @${register ?: statusflag}"
|
"${it.second.type} ${it.second.name} @${register ?: statusflag}"
|
||||||
}.joinToString(", ")
|
}
|
||||||
val clobbers = if (node.clobbers.isEmpty()) "" else "clobbers ${node.clobbers}"
|
val clobbers = if (node.clobbers.isEmpty()) "" else "clobbers ${node.clobbers}"
|
||||||
val returns = if (node.returns.isEmpty()) "" else {
|
val returns = if (node.returns.isEmpty()) "" else {
|
||||||
"-> ${node.returns.map {
|
"-> ${node.returns.joinToString(", ") {
|
||||||
val register = it.first.registerOrPair
|
val register = it.first.registerOrPair
|
||||||
val statusflag = it.first.statusflag
|
val statusflag = it.first.statusflag
|
||||||
"${it.second} @${register ?: statusflag}"}
|
"${it.second} @${register ?: statusflag}"
|
||||||
.joinToString(", ")
|
}
|
||||||
}"
|
}"
|
||||||
}
|
}
|
||||||
val str = if (node.inline) "inline " else ""
|
val str = if (node.inline) "inline " else ""
|
||||||
if(node.address==null) {
|
if(node.address == null) {
|
||||||
str + "asmsub ${node.name}($params) $clobbers $returns"
|
str + "asmsub ${node.name}($params) $clobbers $returns"
|
||||||
} else {
|
} else {
|
||||||
str + "romsub ${node.address.toHex()} = ${node.name}($params) $clobbers $returns"
|
val bank = if(node.address.constbank!=null) "@bank ${node.address.constbank}"
|
||||||
|
else if(node.address.varbank!=null) "@bank ${node.address.varbank?.name}"
|
||||||
|
else ""
|
||||||
|
str + "extsub $bank ${node.address.address.toHex()} = ${node.name}($params) $clobbers $returns"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
is PtBlock -> {
|
is PtBlock -> {
|
||||||
val addr = if(node.options.address==null) "" else "@${node.options.address.toHex()}"
|
val addr = if(node.options.address==null) "" else "@${node.options.address.toHex()}"
|
||||||
val align = if(node.options.alignment==PtBlock.BlockAlignment.NONE) "" else "align=${node.options.alignment}"
|
"\nblock '${node.name}' $addr"
|
||||||
"\nblock '${node.name}' $addr $align"
|
|
||||||
}
|
}
|
||||||
is PtConstant -> {
|
is PtConstant -> {
|
||||||
val value = when(node.type) {
|
val value = when {
|
||||||
DataType.BOOL -> if(node.value==0.0) "false" else "true"
|
node.type.isBool -> if(node.value==0.0) "false" else "true"
|
||||||
in IntegerDatatypes -> node.value.toInt().toString()
|
node.type.isInteger -> node.value.toInt().toString()
|
||||||
else -> node.value.toString()
|
else -> node.value.toString()
|
||||||
}
|
}
|
||||||
"const ${node.type.name.lowercase()} ${node.name} = $value"
|
"const ${node.type} ${node.name} = $value"
|
||||||
}
|
}
|
||||||
is PtLabel -> "${node.name}:"
|
is PtLabel -> "${node.name}:"
|
||||||
is PtMemMapped -> {
|
is PtMemMapped -> {
|
||||||
if(node.type in ArrayDatatypes) {
|
if(node.type.isArray) {
|
||||||
val arraysize = if(node.arraySize==null) "" else node.arraySize.toString()
|
val arraysize = if(node.arraySize==null) "" else node.arraySize.toString()
|
||||||
val eltType = ArrayToElementTypes.getValue(node.type)
|
"&${node.type.elementType()}[$arraysize] ${node.name} = ${node.address.toHex()}"
|
||||||
"&${eltType.name.lowercase()}[$arraysize] ${node.name} = ${node.address.toHex()}"
|
|
||||||
} else {
|
} else {
|
||||||
"&${node.type.name.lowercase()} ${node.name} = ${node.address.toHex()}"
|
"&${node.type} ${node.name} = ${node.address.toHex()}"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
is PtSub -> {
|
is PtSub -> {
|
||||||
val params = node.parameters.map { "${it.type} ${it.name}" }.joinToString(", ")
|
val params = node.parameters.joinToString(", ") {
|
||||||
|
val reg = if(it.register!=null) "@${it.register}" else ""
|
||||||
|
"${it.type} ${it.name} $reg"
|
||||||
|
}
|
||||||
var str = "sub ${node.name}($params) "
|
var str = "sub ${node.name}($params) "
|
||||||
if(node.returntype!=null)
|
if(node.returntype!=null)
|
||||||
str += "-> ${node.returntype.name.lowercase()}"
|
str += "-> ${node.returntype}"
|
||||||
str
|
str
|
||||||
}
|
}
|
||||||
is PtVariable -> {
|
is PtVariable -> {
|
||||||
val split = if(node.type in SplitWordArrayTypes) "@split" else ""
|
val split = if(node.type.isSplitWordArray) "" else "@nosplit"
|
||||||
val str = if(node.arraySize!=null) {
|
val align = when(node.align) {
|
||||||
val eltType = ArrayToElementTypes.getValue(node.type)
|
0u -> ""
|
||||||
"${eltType.name.lowercase()}[${node.arraySize}] $split ${node.name}"
|
2u -> "@alignword"
|
||||||
|
64u -> "@align64"
|
||||||
|
256u -> "@alignpage"
|
||||||
|
else -> throw IllegalArgumentException("invalid alignment size")
|
||||||
}
|
}
|
||||||
else if(node.type in ArrayDatatypes) {
|
val str = if(node.arraySize!=null) {
|
||||||
val eltType = ArrayToElementTypes.getValue(node.type)
|
val eltType = node.type.elementType()
|
||||||
"${eltType.name.lowercase()}[] $split ${node.name}"
|
"${eltType}[${node.arraySize}] $split $align ${node.name}"
|
||||||
|
}
|
||||||
|
else if(node.type.isArray) {
|
||||||
|
val eltType = node.type.elementType()
|
||||||
|
"${eltType}[] $split $align ${node.name}"
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
"${node.type.name.lowercase()} ${node.name}"
|
"${node.type} ${node.name}"
|
||||||
if(node.value!=null)
|
if(node.value!=null)
|
||||||
str + " = " + txt(node.value)
|
str + " = " + txt(node.value)
|
||||||
else
|
else
|
||||||
@ -136,7 +158,10 @@ fun printAst(root: PtNode, skipLibraries: Boolean, output: (text: String) -> Uni
|
|||||||
is PtProgram -> "PROGRAM ${node.name}"
|
is PtProgram -> "PROGRAM ${node.name}"
|
||||||
is PtRepeatLoop -> "repeat"
|
is PtRepeatLoop -> "repeat"
|
||||||
is PtReturn -> "return"
|
is PtReturn -> "return"
|
||||||
is PtSubroutineParameter -> "${node.type.name.lowercase()} ${node.name}"
|
is PtSubroutineParameter -> {
|
||||||
|
val reg = if(node.register!=null) "@${node.register}" else ""
|
||||||
|
"${node.type} ${node.name} $reg"
|
||||||
|
}
|
||||||
is PtWhen -> "when"
|
is PtWhen -> "when"
|
||||||
is PtWhenChoice -> {
|
is PtWhenChoice -> {
|
||||||
if(node.isElse)
|
if(node.isElse)
|
||||||
@ -144,6 +169,8 @@ fun printAst(root: PtNode, skipLibraries: Boolean, output: (text: String) -> Uni
|
|||||||
else
|
else
|
||||||
"->"
|
"->"
|
||||||
}
|
}
|
||||||
|
is PtDefer -> "<defer>"
|
||||||
|
is PtIfExpression -> "<ifexpr>"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,17 +5,21 @@ import prog8.code.core.*
|
|||||||
|
|
||||||
sealed interface IPtSubroutine {
|
sealed interface IPtSubroutine {
|
||||||
val name: String
|
val name: String
|
||||||
|
val scopedName: String
|
||||||
}
|
}
|
||||||
|
|
||||||
class PtAsmSub(
|
class PtAsmSub(
|
||||||
name: String,
|
name: String,
|
||||||
val address: UInt?,
|
val address: Address?,
|
||||||
val clobbers: Set<CpuRegister>,
|
val clobbers: Set<CpuRegister>,
|
||||||
val parameters: List<Pair<RegisterOrStatusflag, PtSubroutineParameter>>,
|
val parameters: List<Pair<RegisterOrStatusflag, PtSubroutineParameter>>,
|
||||||
val returns: List<Pair<RegisterOrStatusflag, DataType>>,
|
val returns: List<Pair<RegisterOrStatusflag, DataType>>,
|
||||||
val inline: Boolean,
|
val inline: Boolean,
|
||||||
position: Position
|
position: Position
|
||||||
) : PtNamedNode(name, position), IPtSubroutine
|
) : PtNamedNode(name, position), IPtSubroutine {
|
||||||
|
|
||||||
|
class Address(val constbank: UByte?, var varbank: PtIdentifier?, val address: UInt)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtSub(
|
class PtSub(
|
||||||
@ -26,24 +30,33 @@ class PtSub(
|
|||||||
) : PtNamedNode(name, position), IPtSubroutine, IPtStatementContainer {
|
) : PtNamedNode(name, position), IPtSubroutine, IPtStatementContainer {
|
||||||
init {
|
init {
|
||||||
// params and return value should not be str
|
// params and return value should not be str
|
||||||
if(parameters.any{ it.type !in NumericDatatypes && it.type!=DataType.BOOL })
|
if(parameters.any{ !it.type.isNumericOrBool })
|
||||||
throw AssemblyError("non-numeric/non-bool parameter")
|
throw AssemblyError("non-numeric/non-bool parameter")
|
||||||
if(returntype!=null && returntype !in NumericDatatypes && returntype!=DataType.BOOL)
|
if(returntype!=null && !returntype.isNumericOrBool)
|
||||||
throw AssemblyError("non-numeric/non-bool returntype $returntype")
|
throw AssemblyError("non-numeric/non-bool returntype $returntype")
|
||||||
parameters.forEach { it.parent=this }
|
parameters.forEach { it.parent=this }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtSubroutineParameter(name: String, val type: DataType, position: Position): PtNamedNode(name, position)
|
class PtSubroutineParameter(name: String, val type: DataType, val register: RegisterOrPair?, position: Position): PtNamedNode(name, position)
|
||||||
|
|
||||||
|
|
||||||
sealed interface IPtAssignment {
|
sealed interface IPtAssignment {
|
||||||
val children: MutableList<PtNode>
|
val children: MutableList<PtNode>
|
||||||
val target: PtAssignTarget
|
val target: PtAssignTarget
|
||||||
get() = children[0] as PtAssignTarget
|
get() {
|
||||||
|
if(children.size==2)
|
||||||
|
return children[0] as PtAssignTarget
|
||||||
|
else if(children.size<2)
|
||||||
|
throw AssemblyError("incomplete node")
|
||||||
|
else
|
||||||
|
throw AssemblyError("no singular target")
|
||||||
|
}
|
||||||
val value: PtExpression
|
val value: PtExpression
|
||||||
get() = children[1] as PtExpression
|
get() = children.last() as PtExpression
|
||||||
|
val multiTarget: Boolean
|
||||||
|
get() = children.size>2
|
||||||
}
|
}
|
||||||
|
|
||||||
class PtAssignment(position: Position) : PtNode(position), IPtAssignment
|
class PtAssignment(position: Position) : PtNode(position), IPtAssignment
|
||||||
@ -51,7 +64,7 @@ class PtAssignment(position: Position) : PtNode(position), IPtAssignment
|
|||||||
class PtAugmentedAssign(val operator: String, position: Position) : PtNode(position), IPtAssignment
|
class PtAugmentedAssign(val operator: String, position: Position) : PtNode(position), IPtAssignment
|
||||||
|
|
||||||
|
|
||||||
class PtAssignTarget(position: Position) : PtNode(position) {
|
class PtAssignTarget(val void: Boolean, position: Position) : PtNode(position) {
|
||||||
val identifier: PtIdentifier?
|
val identifier: PtIdentifier?
|
||||||
get() = children.single() as? PtIdentifier
|
get() = children.single() as? PtIdentifier
|
||||||
val array: PtArrayIndexer?
|
val array: PtArrayIndexer?
|
||||||
@ -69,7 +82,7 @@ class PtAssignTarget(position: Position) : PtNode(position) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
infix fun isSameAs(expression: PtExpression): Boolean = expression.isSameAs(this)
|
infix fun isSameAs(expression: PtExpression): Boolean = !void && expression.isSameAs(this)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -103,12 +116,9 @@ class PtIfElse(position: Position) : PtNode(position) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtJump(val identifier: PtIdentifier?, // note: even ad-hoc labels are wrapped as an Identifier to simplify code. Just use dummy type and position.
|
class PtJump(position: Position) : PtNode(position) {
|
||||||
val address: UInt?,
|
val target: PtExpression
|
||||||
position: Position) : PtNode(position) {
|
get() = children.single() as PtExpression
|
||||||
init {
|
|
||||||
identifier?.let {it.parent = this }
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -121,7 +131,8 @@ class PtRepeatLoop(position: Position) : PtNode(position) {
|
|||||||
|
|
||||||
|
|
||||||
class PtReturn(position: Position) : PtNode(position) {
|
class PtReturn(position: Position) : PtNode(position) {
|
||||||
val hasValue = children.any()
|
val hasValue: Boolean
|
||||||
|
get() = children.any()
|
||||||
val value: PtExpression?
|
val value: PtExpression?
|
||||||
get() {
|
get() {
|
||||||
return if(children.any())
|
return if(children.any())
|
||||||
@ -138,7 +149,15 @@ sealed interface IPtVariable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
class PtVariable(name: String, override val type: DataType, val zeropage: ZeropageWish, val value: PtExpression?, val arraySize: UInt?, position: Position) : PtNamedNode(name, position), IPtVariable {
|
class PtVariable(
|
||||||
|
name: String,
|
||||||
|
override val type: DataType,
|
||||||
|
val zeropage: ZeropageWish,
|
||||||
|
val align: UInt,
|
||||||
|
val value: PtExpression?,
|
||||||
|
val arraySize: UInt?,
|
||||||
|
position: Position
|
||||||
|
) : PtNamedNode(name, position), IPtVariable {
|
||||||
init {
|
init {
|
||||||
value?.let {it.parent=this}
|
value?.let {it.parent=this}
|
||||||
}
|
}
|
||||||
@ -150,7 +169,7 @@ class PtConstant(name: String, override val type: DataType, val value: Double, p
|
|||||||
|
|
||||||
class PtMemMapped(name: String, override val type: DataType, val address: UInt, val arraySize: UInt?, position: Position) : PtNamedNode(name, position), IPtVariable {
|
class PtMemMapped(name: String, override val type: DataType, val address: UInt, val arraySize: UInt?, position: Position) : PtNamedNode(name, position), IPtVariable {
|
||||||
init {
|
init {
|
||||||
require(type!=DataType.BOOL && type!=DataType.ARRAY_BOOL)
|
require(!type.isString)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -169,3 +188,6 @@ class PtWhenChoice(val isElse: Boolean, position: Position) : PtNode(position) {
|
|||||||
val statements: PtNodeGroup
|
val statements: PtNodeGroup
|
||||||
get() = children[1] as PtNodeGroup
|
get() = children[1] as PtNodeGroup
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class PtDefer(position: Position): PtNode(position), IPtStatementContainer
|
||||||
|
16
codeCore/src/prog8/code/ast/Verify.kt
Normal file
16
codeCore/src/prog8/code/ast/Verify.kt
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
package prog8.code.ast
|
||||||
|
|
||||||
|
import prog8.code.SymbolTable
|
||||||
|
import prog8.code.core.*
|
||||||
|
|
||||||
|
fun verifyFinalAstBeforeAsmGen(program: PtProgram, options: CompilationOptions, st: SymbolTable, errors: IErrorReporter) {
|
||||||
|
/*
|
||||||
|
walkAst(program) { node, _ ->
|
||||||
|
if(node is PtVariable) {
|
||||||
|
if(node.value!=null) {
|
||||||
|
// require(node.type in ArrayDatatypes || node.type==DataType.STR) { "final check before asmgen: only string and array variables can still have an init value ${node.name} ${node.type} ${node.position}"}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
}
|
@ -1,7 +1,7 @@
|
|||||||
package prog8.code.core
|
package prog8.code.core
|
||||||
|
|
||||||
class ReturnConvention(val dt: DataType?, val reg: RegisterOrPair?, val floatFac1: Boolean)
|
class ReturnConvention(val dt: BaseDataType?, val reg: RegisterOrPair?)
|
||||||
class ParamConvention(val dt: DataType, val reg: RegisterOrPair?, val variable: Boolean)
|
class ParamConvention(val dt: BaseDataType, val reg: RegisterOrPair?, val variable: Boolean)
|
||||||
class CallConvention(val params: List<ParamConvention>, val returns: ReturnConvention) {
|
class CallConvention(val params: List<ParamConvention>, val returns: ReturnConvention) {
|
||||||
override fun toString(): String {
|
override fun toString(): String {
|
||||||
val paramConvs = params.mapIndexed { index, it ->
|
val paramConvs = params.mapIndexed { index, it ->
|
||||||
@ -13,35 +13,43 @@ class CallConvention(val params: List<ParamConvention>, val returns: ReturnConve
|
|||||||
}
|
}
|
||||||
val returnConv =
|
val returnConv =
|
||||||
when {
|
when {
|
||||||
returns.reg!=null -> returns.reg.toString()
|
returns.reg == RegisterOrPair.FAC1 -> "floatFAC1"
|
||||||
returns.floatFac1 -> "floatFAC1"
|
returns.reg != null -> returns.reg.toString()
|
||||||
else -> "<no returnvalue>"
|
else -> "<no returnvalue>"
|
||||||
}
|
}
|
||||||
return "CallConvention[" + paramConvs.joinToString() + " ; returns: $returnConv]"
|
return "CallConvention[" + paramConvs.joinToString() + " ; returns: $returnConv]"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class FParam(val name: String, val possibleDatatypes: Array<DataType>)
|
class FParam(val name: String, vararg val possibleDatatypes: BaseDataType)
|
||||||
|
|
||||||
|
|
||||||
|
private val IterableDatatypes = arrayOf(BaseDataType.STR, BaseDataType.ARRAY, BaseDataType.ARRAY_SPLITW)
|
||||||
|
private val IntegerDatatypes = arrayOf(BaseDataType.UBYTE, BaseDataType.BYTE, BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)
|
||||||
|
private val NumericDatatypes = arrayOf(BaseDataType.UBYTE, BaseDataType.BYTE, BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG, BaseDataType.FLOAT)
|
||||||
|
private val ByteDatatypes = arrayOf(BaseDataType.UBYTE, BaseDataType.BYTE)
|
||||||
|
private val ArrayDatatypes = arrayOf(BaseDataType.ARRAY, BaseDataType.ARRAY_SPLITW)
|
||||||
|
|
||||||
|
|
||||||
class FSignature(val pure: Boolean, // does it have side effects?
|
class FSignature(val pure: Boolean, // does it have side effects?
|
||||||
val parameters: List<FParam>,
|
val returnType: BaseDataType?,
|
||||||
val returnType: DataType?) {
|
vararg val parameters: FParam) {
|
||||||
|
|
||||||
fun callConvention(actualParamTypes: List<DataType>): CallConvention {
|
fun callConvention(actualParamTypes: List<BaseDataType>): CallConvention {
|
||||||
val returns: ReturnConvention = when (returnType) {
|
val returns: ReturnConvention = when (returnType) {
|
||||||
DataType.UBYTE, DataType.BYTE -> ReturnConvention(returnType, RegisterOrPair.A, false)
|
BaseDataType.UBYTE, BaseDataType.BYTE -> ReturnConvention(returnType, RegisterOrPair.A)
|
||||||
DataType.UWORD, DataType.WORD -> ReturnConvention(returnType, RegisterOrPair.AY, false)
|
BaseDataType.UWORD, BaseDataType.WORD -> ReturnConvention(returnType, RegisterOrPair.AY)
|
||||||
DataType.FLOAT -> ReturnConvention(returnType, null, true)
|
BaseDataType.FLOAT -> ReturnConvention(returnType, RegisterOrPair.FAC1)
|
||||||
in PassByReferenceDatatypes -> ReturnConvention(returnType!!, RegisterOrPair.AY, false)
|
in IterableDatatypes -> ReturnConvention(returnType!!, RegisterOrPair.AY)
|
||||||
null -> ReturnConvention(null, null, false)
|
null -> ReturnConvention(null, null)
|
||||||
else -> {
|
else -> {
|
||||||
// return type depends on arg type
|
// return type depends on arg type
|
||||||
when (val paramType = actualParamTypes.first()) {
|
when (val paramType = actualParamTypes.first()) {
|
||||||
DataType.UBYTE, DataType.BYTE -> ReturnConvention(paramType, RegisterOrPair.A, false)
|
BaseDataType.UBYTE, BaseDataType.BYTE -> ReturnConvention(paramType, RegisterOrPair.A)
|
||||||
DataType.UWORD, DataType.WORD -> ReturnConvention(paramType, RegisterOrPair.AY, false)
|
BaseDataType.UWORD, BaseDataType.WORD -> ReturnConvention(paramType, RegisterOrPair.AY)
|
||||||
DataType.FLOAT -> ReturnConvention(paramType, null, true)
|
BaseDataType.FLOAT -> ReturnConvention(paramType, RegisterOrPair.FAC1)
|
||||||
in PassByReferenceDatatypes -> ReturnConvention(paramType, RegisterOrPair.AY, false)
|
in IterableDatatypes -> ReturnConvention(paramType, RegisterOrPair.AY)
|
||||||
else -> ReturnConvention(paramType, null, false)
|
else -> ReturnConvention(paramType, null)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -49,16 +57,27 @@ class FSignature(val pure: Boolean, // does it have side effects?
|
|||||||
return when {
|
return when {
|
||||||
actualParamTypes.isEmpty() -> CallConvention(emptyList(), returns)
|
actualParamTypes.isEmpty() -> CallConvention(emptyList(), returns)
|
||||||
actualParamTypes.size==1 -> {
|
actualParamTypes.size==1 -> {
|
||||||
// one parameter goes via register/registerpair
|
// One parameter goes via register/registerpair.
|
||||||
|
// this avoids repeated code for every caller to store the value in the subroutine's argument variable.
|
||||||
|
// (that store is still done, but only coded once at the start at the subroutine itself rather than at every call site).
|
||||||
val paramConv = when(val paramType = actualParamTypes[0]) {
|
val paramConv = when(val paramType = actualParamTypes[0]) {
|
||||||
DataType.UBYTE, DataType.BYTE -> ParamConvention(paramType, RegisterOrPair.A, false)
|
BaseDataType.UBYTE, BaseDataType.BYTE -> ParamConvention(paramType, RegisterOrPair.A, false)
|
||||||
DataType.UWORD, DataType.WORD -> ParamConvention(paramType, RegisterOrPair.AY, false)
|
BaseDataType.UWORD, BaseDataType.WORD -> ParamConvention(paramType, RegisterOrPair.AY, false)
|
||||||
DataType.FLOAT -> ParamConvention(paramType, RegisterOrPair.AY, false)
|
BaseDataType.FLOAT -> ParamConvention(paramType, RegisterOrPair.AY, false) // NOTE: for builtin functions, floating point arguments are passed by reference (so you get a pointer in AY)
|
||||||
in PassByReferenceDatatypes -> ParamConvention(paramType, RegisterOrPair.AY, false)
|
in IterableDatatypes -> ParamConvention(paramType, RegisterOrPair.AY, false)
|
||||||
else -> ParamConvention(paramType, null, false)
|
else -> ParamConvention(paramType, null, false)
|
||||||
}
|
}
|
||||||
CallConvention(listOf(paramConv), returns)
|
CallConvention(listOf(paramConv), returns)
|
||||||
}
|
}
|
||||||
|
actualParamTypes.size==2 && (actualParamTypes[0] in ByteDatatypes && actualParamTypes[1].isWord) -> {
|
||||||
|
TODO("opportunity to pass word+byte arguments in A,Y and X registers but not implemented yet")
|
||||||
|
}
|
||||||
|
actualParamTypes.size==2 && (actualParamTypes[0].isWord && actualParamTypes[1].isByte) -> {
|
||||||
|
TODO("opportunity to pass word+byte arguments in A,Y and X registers but not implemented yet")
|
||||||
|
}
|
||||||
|
actualParamTypes.size==3 && actualParamTypes.all { it.isByte } -> {
|
||||||
|
TODO("opportunity to pass 3 byte arguments in A,Y and X registers but not implemented yet")
|
||||||
|
}
|
||||||
else -> {
|
else -> {
|
||||||
// multiple parameters go via variables
|
// multiple parameters go via variables
|
||||||
val paramConvs = actualParamTypes.map { ParamConvention(it, null, true) }
|
val paramConvs = actualParamTypes.map { ParamConvention(it, null, true) }
|
||||||
@ -69,68 +88,69 @@ class FSignature(val pure: Boolean, // does it have side effects?
|
|||||||
}
|
}
|
||||||
|
|
||||||
val BuiltinFunctions: Map<String, FSignature> = mapOf(
|
val BuiltinFunctions: Map<String, FSignature> = mapOf(
|
||||||
// this set of function have no return value and operate in-place:
|
"setlsb" to FSignature(false, null, FParam("variable", BaseDataType.WORD, BaseDataType.UWORD), FParam("value", BaseDataType.BYTE, BaseDataType.UBYTE)),
|
||||||
"setlsb" to FSignature(false, listOf(FParam("variable", arrayOf(DataType.WORD, DataType.UWORD)), FParam("value", arrayOf(DataType.BYTE, DataType.UBYTE))), null),
|
"setmsb" to FSignature(false, null, FParam("variable", BaseDataType.WORD, BaseDataType.UWORD), FParam("value", BaseDataType.BYTE, BaseDataType.UBYTE)),
|
||||||
"setmsb" to FSignature(false, listOf(FParam("variable", arrayOf(DataType.WORD, DataType.UWORD)), FParam("value", arrayOf(DataType.BYTE, DataType.UBYTE))), null),
|
"rol" to FSignature(false, null, FParam("item", BaseDataType.UBYTE, BaseDataType.UWORD)),
|
||||||
"rol" to FSignature(false, listOf(FParam("item", arrayOf(DataType.UBYTE, DataType.UWORD))), null),
|
"ror" to FSignature(false, null, FParam("item", BaseDataType.UBYTE, BaseDataType.UWORD)),
|
||||||
"ror" to FSignature(false, listOf(FParam("item", arrayOf(DataType.UBYTE, DataType.UWORD))), null),
|
"rol2" to FSignature(false, null, FParam("item", BaseDataType.UBYTE, BaseDataType.UWORD)),
|
||||||
"rol2" to FSignature(false, listOf(FParam("item", arrayOf(DataType.UBYTE, DataType.UWORD))), null),
|
"ror2" to FSignature(false, null, FParam("item", BaseDataType.UBYTE, BaseDataType.UWORD)),
|
||||||
"ror2" to FSignature(false, listOf(FParam("item", arrayOf(DataType.UBYTE, DataType.UWORD))), null),
|
"cmp" to FSignature(false, null, FParam("value1", *IntegerDatatypes), FParam("value2", *NumericDatatypes)), // cmp returns a status in the carry flag, but not a proper return value
|
||||||
"sort" to FSignature(false, listOf(FParam("array", ArrayDatatypes)), null),
|
"prog8_lib_stringcompare" to FSignature(true, BaseDataType.BYTE, FParam("str1", BaseDataType.STR), FParam("str2", BaseDataType.STR)),
|
||||||
"reverse" to FSignature(false, listOf(FParam("array", ArrayDatatypes)), null),
|
"prog8_lib_square_byte" to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.BYTE, BaseDataType.UBYTE)),
|
||||||
// cmp returns a status in the carry flag, but not a proper return value
|
"prog8_lib_square_word" to FSignature(true, BaseDataType.UWORD, FParam("value", BaseDataType.WORD, BaseDataType.UWORD)),
|
||||||
"cmp" to FSignature(false, listOf(FParam("value1", IntegerDatatypes), FParam("value2", NumericDatatypes)), null),
|
"prog8_ifelse_bittest_set" to FSignature(true, BaseDataType.BOOL, FParam("variable", *ByteDatatypes), FParam("bitnumber", BaseDataType.UBYTE)),
|
||||||
"prog8_lib_stringcompare" to FSignature(true, listOf(FParam("str1", arrayOf(DataType.STR)), FParam("str2", arrayOf(DataType.STR))), DataType.BYTE),
|
"prog8_ifelse_bittest_notset" to FSignature(true, BaseDataType.BOOL, FParam("variable", *ByteDatatypes), FParam("bitnumber", BaseDataType.UBYTE)),
|
||||||
"prog8_lib_arraycopy" to FSignature(false, listOf(FParam("source", ArrayDatatypes), FParam("target", ArrayDatatypes)), null),
|
"abs" to FSignature(true, null, FParam("value", *NumericDatatypes)),
|
||||||
"prog8_lib_square_byte" to FSignature(true, listOf(FParam("value", arrayOf(DataType.BYTE, DataType.UBYTE))), DataType.UBYTE),
|
"abs__byte" to FSignature(true, BaseDataType.BYTE, FParam("value", BaseDataType.BYTE)),
|
||||||
"prog8_lib_square_word" to FSignature(true, listOf(FParam("value", arrayOf(DataType.WORD, DataType.UWORD))), DataType.UWORD),
|
"abs__word" to FSignature(true, BaseDataType.WORD, FParam("value", BaseDataType.WORD)),
|
||||||
"abs" to FSignature(true, listOf(FParam("value", NumericDatatypes)), null),
|
"abs__float" to FSignature(true, BaseDataType.FLOAT, FParam("value", BaseDataType.FLOAT)),
|
||||||
"abs__byte" to FSignature(true, listOf(FParam("value", arrayOf(DataType.BYTE))), DataType.BYTE),
|
"len" to FSignature(true, BaseDataType.UWORD, FParam("values", *IterableDatatypes)),
|
||||||
"abs__word" to FSignature(true, listOf(FParam("value", arrayOf(DataType.WORD))), DataType.WORD),
|
"sizeof" to FSignature(true, BaseDataType.UBYTE, FParam("object", *BaseDataType.entries.toTypedArray())),
|
||||||
"abs__float" to FSignature(true, listOf(FParam("value", arrayOf(DataType.FLOAT))), DataType.FLOAT),
|
"sgn" to FSignature(true, BaseDataType.BYTE, FParam("value", *NumericDatatypes)),
|
||||||
"len" to FSignature(true, listOf(FParam("values", IterableDatatypes)), DataType.UWORD),
|
"sqrt" to FSignature(true, null, FParam("value", *NumericDatatypes)),
|
||||||
// normal functions follow:
|
"sqrt__ubyte" to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UBYTE)),
|
||||||
"sizeof" to FSignature(true, listOf(FParam("object", DataType.entries.toTypedArray())), DataType.UBYTE),
|
"sqrt__uword" to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UWORD)),
|
||||||
"sgn" to FSignature(true, listOf(FParam("value", NumericDatatypes)), DataType.BYTE),
|
"sqrt__float" to FSignature(true, BaseDataType.FLOAT, FParam("value", BaseDataType.FLOAT)),
|
||||||
"sqrt" to FSignature(true, listOf(FParam("value", NumericDatatypes)), null),
|
"divmod" to FSignature(false, null, FParam("dividend", BaseDataType.UBYTE, BaseDataType.UWORD), FParam("divisor", BaseDataType.UBYTE, BaseDataType.UWORD), FParam("quotient", BaseDataType.UBYTE, BaseDataType.UWORD), FParam("remainder", BaseDataType.UBYTE, BaseDataType.UWORD)),
|
||||||
"sqrt__ubyte" to FSignature(true, listOf(FParam("value", arrayOf(DataType.UBYTE))), DataType.UBYTE),
|
"divmod__ubyte" to FSignature(false, null, FParam("dividend", BaseDataType.UBYTE), FParam("divisor", BaseDataType.UBYTE), FParam("quotient", BaseDataType.UBYTE), FParam("remainder", BaseDataType.UBYTE)),
|
||||||
"sqrt__uword" to FSignature(true, listOf(FParam("value", arrayOf(DataType.UWORD))), DataType.UBYTE),
|
"divmod__uword" to FSignature(false, null, FParam("dividend", BaseDataType.UWORD), FParam("divisor", BaseDataType.UWORD), FParam("quotient", BaseDataType.UWORD), FParam("remainder", BaseDataType.UWORD)),
|
||||||
"sqrt__float" to FSignature(true, listOf(FParam("value", arrayOf(DataType.FLOAT))), DataType.FLOAT),
|
"lsb" to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)),
|
||||||
"divmod" to FSignature(false, listOf(FParam("number", arrayOf(DataType.UBYTE, DataType.UWORD)), FParam("divident", arrayOf(DataType.UBYTE, DataType.UWORD)), FParam("division", arrayOf(DataType.UBYTE, DataType.UWORD)), FParam("remainder", arrayOf(DataType.UBYTE, DataType.UWORD))), null),
|
"lsw" to FSignature(true, BaseDataType.UWORD, FParam("value", BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)),
|
||||||
"divmod__ubyte" to FSignature(false, listOf(FParam("number", arrayOf(DataType.UBYTE)), FParam("divident", arrayOf(DataType.UBYTE)), FParam("division", arrayOf(DataType.UBYTE)), FParam("remainder", arrayOf(DataType.UBYTE))), null),
|
"msb" to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)),
|
||||||
"divmod__uword" to FSignature(false, listOf(FParam("number", arrayOf(DataType.UWORD)), FParam("divident", arrayOf(DataType.UWORD)), FParam("division", arrayOf(DataType.UWORD)), FParam("remainder", arrayOf(DataType.UWORD))), null),
|
"msw" to FSignature(true, BaseDataType.UWORD, FParam("value", BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)),
|
||||||
"any" to FSignature(true, listOf(FParam("values", ArrayDatatypes)), DataType.BOOL),
|
"mkword" to FSignature(true, BaseDataType.UWORD, FParam("msb", BaseDataType.UBYTE), FParam("lsb", BaseDataType.UBYTE)),
|
||||||
"all" to FSignature(true, listOf(FParam("values", ArrayDatatypes)), DataType.BOOL),
|
"clamp" to FSignature(true, null, FParam("value", BaseDataType.BYTE), FParam("minimum", BaseDataType.BYTE), FParam("maximum", BaseDataType.BYTE)),
|
||||||
"lsb" to FSignature(true, listOf(FParam("value", arrayOf(DataType.UWORD, DataType.WORD))), DataType.UBYTE),
|
"clamp__byte" to FSignature(true, BaseDataType.BYTE, FParam("value", BaseDataType.BYTE), FParam("minimum", BaseDataType.BYTE), FParam("maximum", BaseDataType.BYTE)),
|
||||||
"msb" to FSignature(true, listOf(FParam("value", arrayOf(DataType.UWORD, DataType.WORD))), DataType.UBYTE),
|
"clamp__ubyte" to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UBYTE), FParam("minimum", BaseDataType.UBYTE), FParam("maximum", BaseDataType.UBYTE)),
|
||||||
"mkword" to FSignature(true, listOf(FParam("msb", arrayOf(DataType.UBYTE)), FParam("lsb", arrayOf(DataType.UBYTE))), DataType.UWORD),
|
"clamp__word" to FSignature(true, BaseDataType.WORD, FParam("value", BaseDataType.WORD), FParam("minimum", BaseDataType.WORD), FParam("maximum", BaseDataType.WORD)),
|
||||||
"clamp" to FSignature(true, listOf(FParam("value", arrayOf(DataType.BYTE)), FParam("minimum", arrayOf(DataType.BYTE)), FParam("maximum", arrayOf(DataType.BYTE))), null),
|
"clamp__uword" to FSignature(true, BaseDataType.UWORD, FParam("value", BaseDataType.UWORD), FParam("minimum", BaseDataType.UWORD), FParam("maximum", BaseDataType.UWORD)),
|
||||||
"clamp__byte" to FSignature(true, listOf(FParam("value", arrayOf(DataType.BYTE)), FParam("minimum", arrayOf(DataType.BYTE)), FParam("maximum", arrayOf(DataType.BYTE))), DataType.BYTE),
|
"min" to FSignature(true, null, FParam("val1", BaseDataType.BYTE), FParam("val2", BaseDataType.BYTE)),
|
||||||
"clamp__ubyte" to FSignature(true, listOf(FParam("value", arrayOf(DataType.UBYTE)), FParam("minimum", arrayOf(DataType.UBYTE)), FParam("maximum", arrayOf(DataType.UBYTE))), DataType.UBYTE),
|
"min__byte" to FSignature(true, BaseDataType.BYTE, FParam("val1", BaseDataType.BYTE), FParam("val2", BaseDataType.BYTE)),
|
||||||
"clamp__word" to FSignature(true, listOf(FParam("value", arrayOf(DataType.WORD)), FParam("minimum", arrayOf(DataType.WORD)), FParam("maximum", arrayOf(DataType.WORD))), DataType.WORD),
|
"min__ubyte" to FSignature(true, BaseDataType.UBYTE, FParam("val1", BaseDataType.UBYTE), FParam("val2", BaseDataType.UBYTE)),
|
||||||
"clamp__uword" to FSignature(true, listOf(FParam("value", arrayOf(DataType.UWORD)), FParam("minimum", arrayOf(DataType.UWORD)), FParam("maximum", arrayOf(DataType.UWORD))), DataType.UWORD),
|
"min__word" to FSignature(true, BaseDataType.WORD, FParam("val1", BaseDataType.WORD), FParam("val2", BaseDataType.WORD)),
|
||||||
"min" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.BYTE)), FParam("val2", arrayOf(DataType.BYTE))), null),
|
"min__uword" to FSignature(true, BaseDataType.UWORD, FParam("val1", BaseDataType.UWORD), FParam("val2", BaseDataType.UWORD)),
|
||||||
"min__byte" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.BYTE)), FParam("val2", arrayOf(DataType.BYTE))), DataType.BYTE),
|
"max" to FSignature(true, null, FParam("val1", BaseDataType.BYTE), FParam("val2", BaseDataType.BYTE)),
|
||||||
"min__ubyte" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.UBYTE)), FParam("val2", arrayOf(DataType.UBYTE))), DataType.UBYTE),
|
"max__byte" to FSignature(true, BaseDataType.BYTE, FParam("val1", BaseDataType.BYTE), FParam("val2", BaseDataType.BYTE)),
|
||||||
"min__word" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.WORD)), FParam("val2", arrayOf(DataType.WORD))), DataType.WORD),
|
"max__ubyte" to FSignature(true, BaseDataType.UBYTE, FParam("val1", BaseDataType.UBYTE), FParam("val2", BaseDataType.UBYTE)),
|
||||||
"min__uword" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.UWORD)), FParam("val2", arrayOf(DataType.UWORD))), DataType.UWORD),
|
"max__word" to FSignature(true, BaseDataType.WORD, FParam("val1", BaseDataType.WORD), FParam("val2", BaseDataType.WORD)),
|
||||||
"max" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.BYTE)), FParam("val2", arrayOf(DataType.BYTE))), null),
|
"max__uword" to FSignature(true, BaseDataType.UWORD, FParam("val1", BaseDataType.UWORD), FParam("val2", BaseDataType.UWORD)),
|
||||||
"max__byte" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.BYTE)), FParam("val2", arrayOf(DataType.BYTE))), DataType.BYTE),
|
"peek" to FSignature(true, BaseDataType.UBYTE, FParam("address", BaseDataType.UWORD)),
|
||||||
"max__ubyte" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.UBYTE)), FParam("val2", arrayOf(DataType.UBYTE))), DataType.UBYTE),
|
"peekw" to FSignature(true, BaseDataType.UWORD, FParam("address", BaseDataType.UWORD)),
|
||||||
"max__word" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.WORD)), FParam("val2", arrayOf(DataType.WORD))), DataType.WORD),
|
"peekf" to FSignature(true, BaseDataType.FLOAT, FParam("address", BaseDataType.UWORD)),
|
||||||
"max__uword" to FSignature(true, listOf(FParam("val1", arrayOf(DataType.UWORD)), FParam("val2", arrayOf(DataType.UWORD))), DataType.UWORD),
|
"poke" to FSignature(false, null, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.UBYTE)),
|
||||||
"peek" to FSignature(true, listOf(FParam("address", arrayOf(DataType.UWORD))), DataType.UBYTE),
|
"pokew" to FSignature(false, null, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.UWORD)),
|
||||||
"peekw" to FSignature(true, listOf(FParam("address", arrayOf(DataType.UWORD))), DataType.UWORD),
|
"pokef" to FSignature(false, null, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.FLOAT)),
|
||||||
"peekf" to FSignature(true, listOf(FParam("address", arrayOf(DataType.UWORD))), DataType.FLOAT),
|
"pokemon" to FSignature(false, BaseDataType.UBYTE, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.UBYTE)),
|
||||||
"poke" to FSignature(false, listOf(FParam("address", arrayOf(DataType.UWORD)), FParam("value", arrayOf(DataType.UBYTE))), null),
|
"rsave" to FSignature(false, null),
|
||||||
"pokew" to FSignature(false, listOf(FParam("address", arrayOf(DataType.UWORD)), FParam("value", arrayOf(DataType.UWORD))), null),
|
"rrestore" to FSignature(false, null),
|
||||||
"pokef" to FSignature(false, listOf(FParam("address", arrayOf(DataType.UWORD)), FParam("value", arrayOf(DataType.FLOAT))), null),
|
"memory" to FSignature(true, BaseDataType.UWORD, FParam("name", BaseDataType.STR), FParam("size", BaseDataType.UWORD), FParam("alignment", BaseDataType.UWORD)),
|
||||||
"pokemon" to FSignature(false, listOf(FParam("address", arrayOf(DataType.UWORD)), FParam("value", arrayOf(DataType.UBYTE))), DataType.UBYTE),
|
"callfar" to FSignature(false, BaseDataType.UWORD, FParam("bank", BaseDataType.UBYTE), FParam("address", BaseDataType.UWORD), FParam("arg", BaseDataType.UWORD)),
|
||||||
"rsave" to FSignature(false, emptyList(), null),
|
"callfar2" to FSignature(false, BaseDataType.UWORD, FParam("bank", BaseDataType.UBYTE), FParam("address", BaseDataType.UWORD), FParam("argA", BaseDataType.UBYTE), FParam("argX", BaseDataType.UBYTE), FParam("argY", BaseDataType.UBYTE), FParam("argC", BaseDataType.BOOL)),
|
||||||
"rrestore" to FSignature(false, emptyList(), null),
|
"call" to FSignature(false, BaseDataType.UWORD, FParam("address", BaseDataType.UWORD)),
|
||||||
"memory" to FSignature(true, listOf(FParam("name", arrayOf(DataType.STR)), FParam("size", arrayOf(DataType.UWORD)), FParam("alignment", arrayOf(DataType.UWORD))), DataType.UWORD),
|
|
||||||
"callfar" to FSignature(false, listOf(FParam("bank", arrayOf(DataType.UBYTE)), FParam("address", arrayOf(DataType.UWORD)), FParam("arg", arrayOf(DataType.UWORD))), DataType.UWORD),
|
|
||||||
"call" to FSignature(false, listOf(FParam("address", arrayOf(DataType.UWORD))), DataType.UWORD),
|
|
||||||
)
|
)
|
||||||
|
|
||||||
val InplaceModifyingBuiltinFunctions = setOf("setlsb", "setmsb", "rol", "ror", "rol2", "ror2", "sort", "reverse")
|
val InplaceModifyingBuiltinFunctions = setOf(
|
||||||
|
"setlsb", "setmsb",
|
||||||
|
"rol", "ror", "rol2", "ror2",
|
||||||
|
"divmod", "divmod__ubyte", "divmod__uword"
|
||||||
|
)
|
||||||
|
@ -14,20 +14,22 @@ class CompilationOptions(val output: OutputType,
|
|||||||
val compTarget: ICompilationTarget,
|
val compTarget: ICompilationTarget,
|
||||||
// these are set later, based on command line arguments or options in the source code:
|
// these are set later, based on command line arguments or options in the source code:
|
||||||
var loadAddress: UInt,
|
var loadAddress: UInt,
|
||||||
|
var memtopAddress: UInt,
|
||||||
var warnSymbolShadowing: Boolean = false,
|
var warnSymbolShadowing: Boolean = false,
|
||||||
var optimize: Boolean = false,
|
var optimize: Boolean = false,
|
||||||
var asmQuiet: Boolean = false,
|
var asmQuiet: Boolean = false,
|
||||||
var asmListfile: Boolean = false,
|
var asmListfile: Boolean = false,
|
||||||
var includeSourcelines: Boolean = false,
|
var includeSourcelines: Boolean = false,
|
||||||
var dumpVariables: Boolean = false,
|
var dumpVariables: Boolean = false,
|
||||||
|
var dumpSymbols: Boolean = false,
|
||||||
var experimentalCodegen: Boolean = false,
|
var experimentalCodegen: Boolean = false,
|
||||||
var varsHighBank: Int? = null,
|
var varsHighBank: Int? = null,
|
||||||
var varsGolden: Boolean = false,
|
var varsGolden: Boolean = false,
|
||||||
var slabsHighBank: Int? = null,
|
var slabsHighBank: Int? = null,
|
||||||
var slabsGolden: Boolean = false,
|
var slabsGolden: Boolean = false,
|
||||||
var splitWordArrays: Boolean = false,
|
var dontSplitWordArrays: Boolean = false,
|
||||||
var strictBool: Boolean = true,
|
|
||||||
var breakpointCpuInstruction: String? = null,
|
var breakpointCpuInstruction: String? = null,
|
||||||
|
var ignoreFootguns: Boolean = false,
|
||||||
var outputDir: Path = Path(""),
|
var outputDir: Path = Path(""),
|
||||||
var symbolDefs: Map<String, String> = emptyMap()
|
var symbolDefs: Map<String, String> = emptyMap()
|
||||||
) {
|
) {
|
||||||
|
@ -1,11 +1,17 @@
|
|||||||
package prog8.code.core
|
package prog8.code.core
|
||||||
|
|
||||||
import kotlin.math.abs
|
import kotlin.math.abs
|
||||||
|
import kotlin.math.pow
|
||||||
|
|
||||||
|
val powersOfTwoFloat = (0..16).map { (2.0).pow(it) }.toTypedArray()
|
||||||
|
val negativePowersOfTwoFloat = powersOfTwoFloat.map { -it }.toTypedArray()
|
||||||
|
val powersOfTwoInt = (0..16).map { 2.0.pow(it).toInt() }.toTypedArray()
|
||||||
|
|
||||||
fun Number.toHex(): String {
|
fun Number.toHex(): String {
|
||||||
// 0..15 -> "0".."15"
|
// 0..15 -> "0".."15"
|
||||||
// 16..255 -> "$10".."$ff"
|
// 16..255 -> "$10".."$ff"
|
||||||
// 256..65536 -> "$0100".."$ffff"
|
// 256..65536 -> "$0100".."$ffff"
|
||||||
|
// larger -> "$12345678"
|
||||||
// negative values are prefixed with '-'.
|
// negative values are prefixed with '-'.
|
||||||
val integer = this.toInt()
|
val integer = this.toInt()
|
||||||
if(integer<0)
|
if(integer<0)
|
||||||
@ -14,7 +20,7 @@ fun Number.toHex(): String {
|
|||||||
in 0 until 16 -> integer.toString()
|
in 0 until 16 -> integer.toString()
|
||||||
in 0 until 0x100 -> "$"+integer.toString(16).padStart(2,'0')
|
in 0 until 0x100 -> "$"+integer.toString(16).padStart(2,'0')
|
||||||
in 0 until 0x10000 -> "$"+integer.toString(16).padStart(4,'0')
|
in 0 until 0x10000 -> "$"+integer.toString(16).padStart(4,'0')
|
||||||
else -> throw IllegalArgumentException("number too large for 16 bits $this")
|
else -> "$"+integer.toString(16).padStart(8,'0')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -22,11 +28,12 @@ fun UInt.toHex(): String {
|
|||||||
// 0..15 -> "0".."15"
|
// 0..15 -> "0".."15"
|
||||||
// 16..255 -> "$10".."$ff"
|
// 16..255 -> "$10".."$ff"
|
||||||
// 256..65536 -> "$0100".."$ffff"
|
// 256..65536 -> "$0100".."$ffff"
|
||||||
|
// larger -> "$12345678"
|
||||||
return when (this) {
|
return when (this) {
|
||||||
in 0u until 16u -> this.toString()
|
in 0u until 16u -> this.toString()
|
||||||
in 0u until 0x100u -> "$"+this.toString(16).padStart(2,'0')
|
in 0u until 0x100u -> "$"+this.toString(16).padStart(2,'0')
|
||||||
in 0u until 0x10000u -> "$"+this.toString(16).padStart(4,'0')
|
in 0u until 0x10000u -> "$"+this.toString(16).padStart(4,'0')
|
||||||
else -> throw IllegalArgumentException("number too large for 16 bits $this")
|
else -> "$"+this.toString(16).padStart(8,'0')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,63 +1,234 @@
|
|||||||
package prog8.code.core
|
package prog8.code.core
|
||||||
|
|
||||||
enum class DataType {
|
import java.util.Objects
|
||||||
|
|
||||||
|
enum class BaseDataType {
|
||||||
UBYTE, // pass by value 8 bits unsigned
|
UBYTE, // pass by value 8 bits unsigned
|
||||||
BYTE, // pass by value 8 bits signed
|
BYTE, // pass by value 8 bits signed
|
||||||
UWORD, // pass by value 16 bits unsigned
|
UWORD, // pass by value 16 bits unsigned
|
||||||
WORD, // pass by value 16 bits signed
|
WORD, // pass by value 16 bits signed
|
||||||
LONG, // pass by value 32 bits signed
|
LONG, // pass by value 32 bits signed
|
||||||
FLOAT, // pass by value machine dependent
|
FLOAT, // pass by value machine dependent
|
||||||
BOOL, // pass by value bit 0 of a 8 bit byte
|
BOOL, // pass by value bit 0 of an 8-bit byte
|
||||||
STR, // pass by reference
|
STR, // pass by reference
|
||||||
ARRAY_UB, // pass by reference
|
ARRAY, // pass by reference, subtype is the element type
|
||||||
ARRAY_B, // pass by reference
|
ARRAY_SPLITW, // pass by reference, split word layout, subtype is the element type (restricted to word types)
|
||||||
ARRAY_UW, // pass by reference
|
|
||||||
ARRAY_UW_SPLIT, // pass by reference, lo/hi byte split
|
|
||||||
ARRAY_W, // pass by reference
|
|
||||||
ARRAY_W_SPLIT, // pass by reference, lo/hi byte split
|
|
||||||
ARRAY_F, // pass by reference
|
|
||||||
ARRAY_BOOL, // pass by reference
|
|
||||||
UNDEFINED;
|
UNDEFINED;
|
||||||
|
|
||||||
/**
|
|
||||||
* is the type assignable to the given other type (perhaps via a typecast) without loss of precision?
|
|
||||||
*/
|
|
||||||
infix fun isAssignableTo(targetType: DataType) =
|
|
||||||
when(this) {
|
|
||||||
BOOL -> targetType == BOOL
|
|
||||||
UBYTE -> targetType.oneOf(UBYTE, WORD, UWORD, LONG, FLOAT)
|
|
||||||
BYTE -> targetType.oneOf(BYTE, WORD, LONG, FLOAT)
|
|
||||||
UWORD -> targetType.oneOf(UWORD, LONG, FLOAT)
|
|
||||||
WORD -> targetType.oneOf(WORD, LONG, FLOAT)
|
|
||||||
LONG -> targetType.oneOf(LONG, FLOAT)
|
|
||||||
FLOAT -> targetType.oneOf(FLOAT)
|
|
||||||
STR -> targetType.oneOf(STR, UWORD)
|
|
||||||
in ArrayDatatypes -> targetType == this
|
|
||||||
else -> false
|
|
||||||
}
|
|
||||||
|
|
||||||
fun oneOf(vararg types: DataType) = this in types
|
fun largerSizeThan(other: BaseDataType) =
|
||||||
|
|
||||||
infix fun largerThan(other: DataType) =
|
|
||||||
when {
|
when {
|
||||||
this == other -> false
|
this == other -> false
|
||||||
this in ByteDatatypesWithBoolean -> false
|
this.isByteOrBool -> false
|
||||||
this in WordDatatypes -> other in ByteDatatypesWithBoolean
|
this.isWord -> other.isByteOrBool
|
||||||
this == LONG -> other in ByteDatatypesWithBoolean+WordDatatypes
|
this == LONG -> other.isByteOrBool || other.isWord
|
||||||
this == STR && other == UWORD || this == UWORD && other == STR -> false
|
this == STR && other == UWORD || this == UWORD && other == STR -> false
|
||||||
|
this.isArray -> other != FLOAT
|
||||||
|
this == STR -> other != FLOAT
|
||||||
else -> true
|
else -> true
|
||||||
}
|
}
|
||||||
|
|
||||||
infix fun equalsSize(other: DataType) =
|
fun equalsSize(other: BaseDataType) =
|
||||||
when {
|
when {
|
||||||
this == other -> true
|
this == other -> true
|
||||||
this in ByteDatatypesWithBoolean -> other in ByteDatatypesWithBoolean
|
this.isByteOrBool -> other.isByteOrBool
|
||||||
this in WordDatatypes -> other in WordDatatypes
|
this.isWord -> other.isWord
|
||||||
this== STR && other== UWORD || this== UWORD && other== STR -> true
|
this == STR && other== UWORD || this== UWORD && other== STR -> true
|
||||||
|
this == STR && other.isArray -> true
|
||||||
|
this.isArray && other == STR -> true
|
||||||
else -> false
|
else -> false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val BaseDataType.isByte get() = this in arrayOf(BaseDataType.UBYTE, BaseDataType.BYTE)
|
||||||
|
val BaseDataType.isByteOrBool get() = this in arrayOf(BaseDataType.UBYTE, BaseDataType.BYTE, BaseDataType.BOOL)
|
||||||
|
val BaseDataType.isWord get() = this in arrayOf(BaseDataType.UWORD, BaseDataType.WORD)
|
||||||
|
val BaseDataType.isInteger get() = this in arrayOf(BaseDataType.UBYTE, BaseDataType.BYTE, BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)
|
||||||
|
val BaseDataType.isIntegerOrBool get() = this in arrayOf(BaseDataType.UBYTE, BaseDataType.BYTE, BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG, BaseDataType.BOOL)
|
||||||
|
val BaseDataType.isNumeric get() = this == BaseDataType.FLOAT || this.isInteger
|
||||||
|
val BaseDataType.isNumericOrBool get() = this == BaseDataType.BOOL || this.isNumeric
|
||||||
|
val BaseDataType.isSigned get() = this in arrayOf(BaseDataType.BYTE, BaseDataType.WORD, BaseDataType.LONG, BaseDataType.FLOAT)
|
||||||
|
val BaseDataType.isArray get() = this == BaseDataType.ARRAY || this == BaseDataType.ARRAY_SPLITW
|
||||||
|
val BaseDataType.isSplitWordArray get() = this == BaseDataType.ARRAY_SPLITW
|
||||||
|
val BaseDataType.isIterable get() = this in arrayOf(BaseDataType.STR, BaseDataType.ARRAY, BaseDataType.ARRAY_SPLITW)
|
||||||
|
val BaseDataType.isPassByRef get() = this.isIterable
|
||||||
|
val BaseDataType.isPassByValue get() = !this.isIterable
|
||||||
|
|
||||||
|
|
||||||
|
class DataType private constructor(val base: BaseDataType, val sub: BaseDataType?) {
|
||||||
|
|
||||||
|
init {
|
||||||
|
if(base.isArray) {
|
||||||
|
require(sub != null)
|
||||||
|
if(base.isSplitWordArray)
|
||||||
|
require(sub == BaseDataType.UWORD || sub == BaseDataType.WORD)
|
||||||
|
}
|
||||||
|
else if(base==BaseDataType.STR)
|
||||||
|
require(sub==BaseDataType.UBYTE) { "string subtype should be ubyte" }
|
||||||
|
else
|
||||||
|
require(sub == null) { "only string and array base types can have a subtype"}
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun equals(other: Any?): Boolean {
|
||||||
|
if (this === other) return true
|
||||||
|
if (other !is DataType) return false
|
||||||
|
return base == other.base && sub == other.sub
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun hashCode(): Int = Objects.hash(base, sub)
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
private val simpletypes = mapOf(
|
||||||
|
BaseDataType.UBYTE to DataType(BaseDataType.UBYTE, null),
|
||||||
|
BaseDataType.BYTE to DataType(BaseDataType.BYTE, null),
|
||||||
|
BaseDataType.UWORD to DataType(BaseDataType.UWORD, null),
|
||||||
|
BaseDataType.WORD to DataType(BaseDataType.WORD, null),
|
||||||
|
BaseDataType.LONG to DataType(BaseDataType.LONG, null),
|
||||||
|
BaseDataType.FLOAT to DataType(BaseDataType.FLOAT, null),
|
||||||
|
BaseDataType.BOOL to DataType(BaseDataType.BOOL, null),
|
||||||
|
BaseDataType.STR to DataType(BaseDataType.STR, BaseDataType.UBYTE),
|
||||||
|
BaseDataType.UNDEFINED to DataType(BaseDataType.UNDEFINED, null)
|
||||||
|
)
|
||||||
|
|
||||||
|
fun forDt(dt: BaseDataType) = simpletypes.getValue(dt)
|
||||||
|
|
||||||
|
fun arrayFor(elementDt: BaseDataType, splitwordarray: Boolean=true): DataType {
|
||||||
|
val actualElementDt = if(elementDt==BaseDataType.STR) BaseDataType.UWORD else elementDt // array of strings is actually just an array of UWORD pointers
|
||||||
|
return if(splitwordarray && actualElementDt.isWord)
|
||||||
|
DataType(BaseDataType.ARRAY_SPLITW, actualElementDt)
|
||||||
|
else {
|
||||||
|
if(actualElementDt.isNumericOrBool && actualElementDt != BaseDataType.LONG)
|
||||||
|
DataType(BaseDataType.ARRAY, actualElementDt)
|
||||||
|
else
|
||||||
|
throw NoSuchElementException("invalid element dt "+elementDt)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun elementToArray(splitwords: Boolean = true): DataType {
|
||||||
|
return if (base == BaseDataType.UWORD || base == BaseDataType.WORD || base == BaseDataType.STR) arrayFor(base, splitwords)
|
||||||
|
else arrayFor(base, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun elementType(): DataType =
|
||||||
|
if(base.isArray || base==BaseDataType.STR)
|
||||||
|
forDt(sub!!)
|
||||||
|
else
|
||||||
|
throw IllegalArgumentException("not an array")
|
||||||
|
|
||||||
|
override fun toString(): String = when(base) {
|
||||||
|
BaseDataType.ARRAY -> {
|
||||||
|
when(sub) {
|
||||||
|
BaseDataType.BOOL -> "bool[]"
|
||||||
|
BaseDataType.FLOAT -> "float[]"
|
||||||
|
BaseDataType.BYTE -> "byte[]"
|
||||||
|
BaseDataType.WORD -> "word[]"
|
||||||
|
BaseDataType.UBYTE -> "ubyte[]"
|
||||||
|
BaseDataType.UWORD -> "uword[]"
|
||||||
|
else -> throw IllegalArgumentException("invalid sub type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BaseDataType.ARRAY_SPLITW -> {
|
||||||
|
when(sub) {
|
||||||
|
BaseDataType.WORD -> "word[] (split)"
|
||||||
|
BaseDataType.UWORD -> "uword[] (split)"
|
||||||
|
else -> throw IllegalArgumentException("invalid sub type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else -> base.name.lowercase()
|
||||||
|
}
|
||||||
|
|
||||||
|
fun sourceString(): String = when (base) {
|
||||||
|
BaseDataType.BOOL -> "bool"
|
||||||
|
BaseDataType.UBYTE -> "ubyte"
|
||||||
|
BaseDataType.BYTE -> "byte"
|
||||||
|
BaseDataType.UWORD -> "uword"
|
||||||
|
BaseDataType.WORD -> "word"
|
||||||
|
BaseDataType.LONG -> "long"
|
||||||
|
BaseDataType.FLOAT -> "float"
|
||||||
|
BaseDataType.STR -> "str"
|
||||||
|
BaseDataType.ARRAY -> {
|
||||||
|
when(sub) {
|
||||||
|
BaseDataType.UBYTE -> "ubyte["
|
||||||
|
BaseDataType.UWORD -> "@nosplit uword["
|
||||||
|
BaseDataType.BOOL -> "bool["
|
||||||
|
BaseDataType.BYTE -> "byte["
|
||||||
|
BaseDataType.WORD -> "@nosplit word["
|
||||||
|
BaseDataType.FLOAT -> "float["
|
||||||
|
else -> throw IllegalArgumentException("invalid sub type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BaseDataType.ARRAY_SPLITW -> {
|
||||||
|
when(sub) {
|
||||||
|
BaseDataType.UWORD -> "uword["
|
||||||
|
BaseDataType.WORD -> "word["
|
||||||
|
else -> throw IllegalArgumentException("invalid sub type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
BaseDataType.UNDEFINED -> throw IllegalArgumentException("wrong dt")
|
||||||
|
}
|
||||||
|
|
||||||
|
// is the type assignable to the given other type (perhaps via a typecast) without loss of precision?
|
||||||
|
infix fun isAssignableTo(targetType: DataType) =
|
||||||
|
when(base) {
|
||||||
|
BaseDataType.BOOL -> targetType.base == BaseDataType.BOOL
|
||||||
|
BaseDataType.UBYTE -> targetType.base in arrayOf(BaseDataType.UBYTE, BaseDataType.WORD, BaseDataType.UWORD, BaseDataType.LONG, BaseDataType.FLOAT)
|
||||||
|
BaseDataType.BYTE -> targetType.base in arrayOf(BaseDataType.BYTE, BaseDataType.WORD, BaseDataType.LONG, BaseDataType.FLOAT)
|
||||||
|
BaseDataType.UWORD -> targetType.base in arrayOf(BaseDataType.UWORD, BaseDataType.LONG, BaseDataType.FLOAT)
|
||||||
|
BaseDataType.WORD -> targetType.base in arrayOf(BaseDataType.WORD, BaseDataType.LONG, BaseDataType.FLOAT)
|
||||||
|
BaseDataType.LONG -> targetType.base in arrayOf(BaseDataType.LONG, BaseDataType.FLOAT)
|
||||||
|
BaseDataType.FLOAT -> targetType.base in arrayOf(BaseDataType.FLOAT)
|
||||||
|
BaseDataType.STR -> targetType.base in arrayOf(BaseDataType.STR, BaseDataType.UWORD)
|
||||||
|
BaseDataType.ARRAY, BaseDataType.ARRAY_SPLITW -> targetType.base in arrayOf(BaseDataType.ARRAY, BaseDataType.ARRAY_SPLITW) && targetType.sub == sub
|
||||||
|
BaseDataType.UNDEFINED -> false
|
||||||
|
}
|
||||||
|
|
||||||
|
fun largerSizeThan(other: DataType): Boolean {
|
||||||
|
if(isArray) throw IllegalArgumentException("cannot compare size of array types")
|
||||||
|
return base.largerSizeThan(other.base)
|
||||||
|
}
|
||||||
|
fun equalsSize(other: DataType): Boolean {
|
||||||
|
if(isArray) throw IllegalArgumentException("cannot compare size of array types")
|
||||||
|
return base.equalsSize(other.base)
|
||||||
|
}
|
||||||
|
|
||||||
|
val isUndefined = base == BaseDataType.UNDEFINED
|
||||||
|
val isByte = base.isByte
|
||||||
|
val isUnsignedByte = base == BaseDataType.UBYTE
|
||||||
|
val isSignedByte = base == BaseDataType.BYTE
|
||||||
|
val isByteOrBool = base.isByteOrBool
|
||||||
|
val isWord = base.isWord
|
||||||
|
val isUnsignedWord = base == BaseDataType.UWORD
|
||||||
|
val isSignedWord = base == BaseDataType.WORD
|
||||||
|
val isInteger = base.isInteger
|
||||||
|
val isIntegerOrBool = base.isIntegerOrBool
|
||||||
|
val isNumeric = base.isNumeric
|
||||||
|
val isNumericOrBool = base.isNumericOrBool
|
||||||
|
val isSigned = base.isSigned
|
||||||
|
val isUnsigned = !base.isSigned
|
||||||
|
val isArray = base.isArray
|
||||||
|
val isBoolArray = base.isArray && sub == BaseDataType.BOOL
|
||||||
|
val isByteArray = base.isArray && (sub == BaseDataType.UBYTE || sub == BaseDataType.BYTE)
|
||||||
|
val isUnsignedByteArray = base.isArray && sub == BaseDataType.UBYTE
|
||||||
|
val isSignedByteArray = base.isArray && sub == BaseDataType.BYTE
|
||||||
|
val isWordArray = base.isArray && (sub == BaseDataType.UWORD || sub == BaseDataType.WORD)
|
||||||
|
val isUnsignedWordArray = base.isArray && sub == BaseDataType.UWORD
|
||||||
|
val isSignedWordArray = base.isArray && sub == BaseDataType.WORD
|
||||||
|
val isFloatArray = base.isArray && sub == BaseDataType.FLOAT
|
||||||
|
val isString = base == BaseDataType.STR
|
||||||
|
val isBool = base == BaseDataType.BOOL
|
||||||
|
val isFloat = base == BaseDataType.FLOAT
|
||||||
|
val isLong = base == BaseDataType.LONG
|
||||||
|
val isStringly = base == BaseDataType.STR || base == BaseDataType.UWORD || (base == BaseDataType.ARRAY && (sub == BaseDataType.UBYTE || sub == BaseDataType.BYTE))
|
||||||
|
val isSplitWordArray = base.isSplitWordArray
|
||||||
|
val isSplitUnsignedWordArray = base.isSplitWordArray && sub == BaseDataType.UWORD
|
||||||
|
val isSplitSignedWordArray = base.isSplitWordArray && sub == BaseDataType.WORD
|
||||||
|
val isIterable = base.isIterable
|
||||||
|
val isPassByRef = base.isPassByRef
|
||||||
|
val isPassByValue = base.isPassByValue
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
enum class CpuRegister {
|
enum class CpuRegister {
|
||||||
A,
|
A,
|
||||||
X,
|
X,
|
||||||
@ -78,7 +249,7 @@ enum class RegisterOrPair {
|
|||||||
R8, R9, R10, R11, R12, R13, R14, R15;
|
R8, R9, R10, R11, R12, R13, R14, R15;
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
val names by lazy { values().map { it.toString()} }
|
val names by lazy { entries.map { it.toString()} }
|
||||||
fun fromCpuRegister(cpu: CpuRegister): RegisterOrPair {
|
fun fromCpuRegister(cpu: CpuRegister): RegisterOrPair {
|
||||||
return when(cpu) {
|
return when(cpu) {
|
||||||
CpuRegister.A -> A
|
CpuRegister.A -> A
|
||||||
@ -95,6 +266,17 @@ enum class RegisterOrPair {
|
|||||||
else -> throw IllegalArgumentException("no cpu hardware register for $this")
|
else -> throw IllegalArgumentException("no cpu hardware register for $this")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fun asScopedNameVirtualReg(type: DataType?): List<String> {
|
||||||
|
require(this in Cx16VirtualRegisters)
|
||||||
|
val suffix = when(type?.base) {
|
||||||
|
BaseDataType.UBYTE, BaseDataType.BOOL -> "L"
|
||||||
|
BaseDataType.BYTE -> "sL"
|
||||||
|
BaseDataType.WORD -> "s"
|
||||||
|
BaseDataType.UWORD, null -> ""
|
||||||
|
else -> throw kotlin.IllegalArgumentException("invalid register param type")
|
||||||
|
}
|
||||||
|
return listOf("cx16", name.lowercase()+suffix)
|
||||||
|
}
|
||||||
} // only used in parameter and return value specs in asm subroutines
|
} // only used in parameter and return value specs in asm subroutines
|
||||||
|
|
||||||
enum class Statusflag {
|
enum class Statusflag {
|
||||||
@ -104,7 +286,7 @@ enum class Statusflag {
|
|||||||
Pn; // don't use
|
Pn; // don't use
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
val names by lazy { values().map { it.toString()} }
|
val names by lazy { entries.map { it.toString()} }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -123,48 +305,6 @@ enum class BranchCondition {
|
|||||||
VC
|
VC
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
val ByteDatatypes = arrayOf(DataType.UBYTE, DataType.BYTE)
|
|
||||||
val ByteDatatypesWithBoolean = ByteDatatypes + DataType.BOOL
|
|
||||||
val WordDatatypes = arrayOf(DataType.UWORD, DataType.WORD)
|
|
||||||
val IntegerDatatypes = arrayOf(DataType.UBYTE, DataType.BYTE, DataType.UWORD, DataType.WORD, DataType.LONG)
|
|
||||||
val IntegerDatatypesWithBoolean = IntegerDatatypes + DataType.BOOL
|
|
||||||
val NumericDatatypes = arrayOf(DataType.UBYTE, DataType.BYTE, DataType.UWORD, DataType.WORD, DataType.LONG, DataType.FLOAT)
|
|
||||||
val NumericDatatypesWithBoolean = NumericDatatypes + DataType.BOOL
|
|
||||||
val SignedDatatypes = arrayOf(DataType.BYTE, DataType.WORD, DataType.LONG, DataType.FLOAT)
|
|
||||||
val ArrayDatatypes = arrayOf(DataType.ARRAY_UB, DataType.ARRAY_B, DataType.ARRAY_UW, DataType.ARRAY_UW_SPLIT, DataType.ARRAY_W, DataType.ARRAY_W_SPLIT, DataType.ARRAY_F, DataType.ARRAY_BOOL)
|
|
||||||
val StringlyDatatypes = arrayOf(DataType.STR, DataType.ARRAY_UB, DataType.ARRAY_B, DataType.UWORD)
|
|
||||||
val SplitWordArrayTypes = arrayOf(DataType.ARRAY_UW_SPLIT, DataType.ARRAY_W_SPLIT)
|
|
||||||
val IterableDatatypes = arrayOf(
|
|
||||||
DataType.STR,
|
|
||||||
DataType.ARRAY_UB, DataType.ARRAY_B,
|
|
||||||
DataType.ARRAY_UW, DataType.ARRAY_W,
|
|
||||||
DataType.ARRAY_UW_SPLIT, DataType.ARRAY_W_SPLIT,
|
|
||||||
DataType.ARRAY_F, DataType.ARRAY_BOOL
|
|
||||||
)
|
|
||||||
val PassByValueDatatypes = NumericDatatypesWithBoolean
|
|
||||||
val PassByReferenceDatatypes = IterableDatatypes
|
|
||||||
val ArrayToElementTypes = mapOf(
|
|
||||||
DataType.STR to DataType.UBYTE,
|
|
||||||
DataType.ARRAY_B to DataType.BYTE,
|
|
||||||
DataType.ARRAY_UB to DataType.UBYTE,
|
|
||||||
DataType.ARRAY_W to DataType.WORD,
|
|
||||||
DataType.ARRAY_UW to DataType.UWORD,
|
|
||||||
DataType.ARRAY_W_SPLIT to DataType.WORD,
|
|
||||||
DataType.ARRAY_UW_SPLIT to DataType.UWORD,
|
|
||||||
DataType.ARRAY_F to DataType.FLOAT,
|
|
||||||
DataType.ARRAY_BOOL to DataType.BOOL
|
|
||||||
)
|
|
||||||
val ElementToArrayTypes = mapOf(
|
|
||||||
DataType.BYTE to DataType.ARRAY_B,
|
|
||||||
DataType.UBYTE to DataType.ARRAY_UB,
|
|
||||||
DataType.WORD to DataType.ARRAY_W,
|
|
||||||
DataType.UWORD to DataType.ARRAY_UW,
|
|
||||||
DataType.FLOAT to DataType.ARRAY_F,
|
|
||||||
DataType.BOOL to DataType.ARRAY_BOOL,
|
|
||||||
DataType.STR to DataType.ARRAY_UW // array of str is just an array of pointers
|
|
||||||
)
|
|
||||||
|
|
||||||
val Cx16VirtualRegisters = arrayOf(
|
val Cx16VirtualRegisters = arrayOf(
|
||||||
RegisterOrPair.R0, RegisterOrPair.R1, RegisterOrPair.R2, RegisterOrPair.R3,
|
RegisterOrPair.R0, RegisterOrPair.R1, RegisterOrPair.R2, RegisterOrPair.R3,
|
||||||
RegisterOrPair.R4, RegisterOrPair.R5, RegisterOrPair.R6, RegisterOrPair.R7,
|
RegisterOrPair.R4, RegisterOrPair.R5, RegisterOrPair.R6, RegisterOrPair.R7,
|
||||||
@ -172,7 +312,7 @@ val Cx16VirtualRegisters = arrayOf(
|
|||||||
RegisterOrPair.R12, RegisterOrPair.R13, RegisterOrPair.R14, RegisterOrPair.R15
|
RegisterOrPair.R12, RegisterOrPair.R13, RegisterOrPair.R14, RegisterOrPair.R15
|
||||||
)
|
)
|
||||||
|
|
||||||
val CpuRegisters = setOf(
|
val CpuRegisters = arrayOf(
|
||||||
RegisterOrPair.A, RegisterOrPair.X, RegisterOrPair.Y,
|
RegisterOrPair.A, RegisterOrPair.X, RegisterOrPair.Y,
|
||||||
RegisterOrPair.AX, RegisterOrPair.AY, RegisterOrPair.XY
|
RegisterOrPair.AX, RegisterOrPair.AY, RegisterOrPair.XY
|
||||||
)
|
)
|
||||||
@ -203,3 +343,9 @@ enum class ZeropageWish {
|
|||||||
DONTCARE,
|
DONTCARE,
|
||||||
NOT_IN_ZEROPAGE
|
NOT_IN_ZEROPAGE
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enum class SplitWish {
|
||||||
|
DONTCARE,
|
||||||
|
SPLIT,
|
||||||
|
NOSPLIT
|
||||||
|
}
|
@ -13,7 +13,9 @@ interface IMachineDefinition {
|
|||||||
val FLOAT_MAX_NEGATIVE: Double
|
val FLOAT_MAX_NEGATIVE: Double
|
||||||
val FLOAT_MAX_POSITIVE: Double
|
val FLOAT_MAX_POSITIVE: Double
|
||||||
val FLOAT_MEM_SIZE: Int
|
val FLOAT_MEM_SIZE: Int
|
||||||
|
val STARTUP_CODE_RESERVED_SIZE: UInt // this is here, so that certain compiler targets are able to tune this
|
||||||
val PROGRAM_LOAD_ADDRESS : UInt
|
val PROGRAM_LOAD_ADDRESS : UInt
|
||||||
|
val PROGRAM_MEMTOP_ADDRESS: UInt
|
||||||
val BSSHIGHRAM_START: UInt
|
val BSSHIGHRAM_START: UInt
|
||||||
val BSSHIGHRAM_END: UInt
|
val BSSHIGHRAM_END: UInt
|
||||||
val BSSGOLDENRAM_START: UInt
|
val BSSGOLDENRAM_START: UInt
|
||||||
@ -26,7 +28,9 @@ interface IMachineDefinition {
|
|||||||
fun initializeMemoryAreas(compilerOptions: CompilationOptions)
|
fun initializeMemoryAreas(compilerOptions: CompilationOptions)
|
||||||
fun getFloatAsmBytes(num: Number): String
|
fun getFloatAsmBytes(num: Number): String
|
||||||
|
|
||||||
fun importLibs(compilerOptions: CompilationOptions, compilationTargetName: String): List<String>
|
fun convertFloatToBytes(num: Double): List<UByte>
|
||||||
|
fun convertBytesToFloat(bytes: List<UByte>): Double
|
||||||
|
|
||||||
fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path)
|
fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path)
|
||||||
fun isIOAddress(address: UInt): Boolean
|
fun isIOAddress(address: UInt): Boolean
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,15 @@
|
|||||||
package prog8.code.core
|
package prog8.code.core
|
||||||
|
|
||||||
interface IMemSizer {
|
interface IMemSizer {
|
||||||
fun memorySize(dt: DataType): Int
|
fun memorySize(dt: DataType, numElements: Int?): Int
|
||||||
fun memorySize(arrayDt: DataType, numElements: Int): Int
|
|
||||||
|
fun memorySize(dt: BaseDataType): Int {
|
||||||
|
if(dt.isPassByRef)
|
||||||
|
return memorySize(DataType.forDt(BaseDataType.UWORD), null) // a pointer size
|
||||||
|
try {
|
||||||
|
return memorySize(DataType.forDt(dt), null)
|
||||||
|
} catch (x: NoSuchElementException) {
|
||||||
|
throw IllegalArgumentException(x.message)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,11 @@ enum class Encoding(val prefix: String) {
|
|||||||
PETSCII("petscii"), // c64/c128/cx16
|
PETSCII("petscii"), // c64/c128/cx16
|
||||||
SCREENCODES("sc"), // c64/c128/cx16
|
SCREENCODES("sc"), // c64/c128/cx16
|
||||||
ATASCII("atascii"), // atari
|
ATASCII("atascii"), // atari
|
||||||
ISO("iso") // cx16
|
ISO("iso"), // cx16 (iso-8859-15)
|
||||||
|
ISO5("iso5"), // cx16 (iso-8859-5, cyrillic)
|
||||||
|
ISO16("iso16"), // cx16 (iso-8859-16, eastern european)
|
||||||
|
CP437("cp437"), // cx16 (ibm pc, codepage 437)
|
||||||
|
KATAKANA("kata") // cx16 (katakana)
|
||||||
}
|
}
|
||||||
|
|
||||||
interface IStringEncoding {
|
interface IStringEncoding {
|
||||||
|
@ -38,7 +38,7 @@ abstract class Zeropage(options: CompilationOptions): MemoryAllocator(options) {
|
|||||||
for (reserved in options.zpReserved)
|
for (reserved in options.zpReserved)
|
||||||
reserve(reserved)
|
reserve(reserved)
|
||||||
|
|
||||||
free.removeAll(setOf(SCRATCH_B1, SCRATCH_REG, SCRATCH_W1, SCRATCH_W1 + 1u, SCRATCH_W2, SCRATCH_W2 + 1u))
|
free.removeAll(arrayOf(SCRATCH_B1, SCRATCH_REG, SCRATCH_W1, SCRATCH_W1 + 1u, SCRATCH_W2, SCRATCH_W2 + 1u))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -70,9 +70,9 @@ abstract class Zeropage(options: CompilationOptions): MemoryAllocator(options) {
|
|||||||
return Err(MemAllocationError("zero page usage has been disabled"))
|
return Err(MemAllocationError("zero page usage has been disabled"))
|
||||||
|
|
||||||
val size: Int =
|
val size: Int =
|
||||||
when (datatype) {
|
when {
|
||||||
in IntegerDatatypesWithBoolean -> options.compTarget.memorySize(datatype)
|
datatype.isIntegerOrBool -> options.compTarget.memorySize(datatype, null)
|
||||||
DataType.STR, in ArrayDatatypes -> {
|
datatype.isString || datatype.isArray -> {
|
||||||
val memsize = options.compTarget.memorySize(datatype, numElements!!)
|
val memsize = options.compTarget.memorySize(datatype, numElements!!)
|
||||||
if(position!=null)
|
if(position!=null)
|
||||||
errors.warn("allocating a large value in zeropage; str/array $memsize bytes", position)
|
errors.warn("allocating a large value in zeropage; str/array $memsize bytes", position)
|
||||||
@ -80,9 +80,9 @@ abstract class Zeropage(options: CompilationOptions): MemoryAllocator(options) {
|
|||||||
errors.warn("$name: allocating a large value in zeropage; str/array $memsize bytes", Position.DUMMY)
|
errors.warn("$name: allocating a large value in zeropage; str/array $memsize bytes", Position.DUMMY)
|
||||||
memsize
|
memsize
|
||||||
}
|
}
|
||||||
DataType.FLOAT -> {
|
datatype.isFloat -> {
|
||||||
if (options.floats) {
|
if (options.floats) {
|
||||||
val memsize = options.compTarget.memorySize(DataType.FLOAT)
|
val memsize = options.compTarget.memorySize(DataType.forDt(BaseDataType.FLOAT), null)
|
||||||
if(position!=null)
|
if(position!=null)
|
||||||
errors.warn("allocating a large value in zeropage; float $memsize bytes", position)
|
errors.warn("allocating a large value in zeropage; float $memsize bytes", position)
|
||||||
else
|
else
|
||||||
@ -94,7 +94,7 @@ abstract class Zeropage(options: CompilationOptions): MemoryAllocator(options) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
synchronized(this) {
|
synchronized(this) {
|
||||||
if(free.size > 0) {
|
if(free.isNotEmpty()) {
|
||||||
if(size==1) {
|
if(size==1) {
|
||||||
for(candidate in free.minOrNull()!! .. free.maxOrNull()!!+1u) {
|
for(candidate in free.minOrNull()!! .. free.maxOrNull()!!+1u) {
|
||||||
if(oneSeparateByteFree(candidate))
|
if(oneSeparateByteFree(candidate))
|
||||||
@ -118,10 +118,10 @@ abstract class Zeropage(options: CompilationOptions): MemoryAllocator(options) {
|
|||||||
require(size>=0)
|
require(size>=0)
|
||||||
free.removeAll(address until address+size.toUInt())
|
free.removeAll(address until address+size.toUInt())
|
||||||
if(name.isNotEmpty()) {
|
if(name.isNotEmpty()) {
|
||||||
allocatedVariables[name] = when(datatype) {
|
allocatedVariables[name] = when {
|
||||||
in NumericDatatypes, DataType.BOOL -> VarAllocation(address, datatype, size) // numerical variables in zeropage never have an initial value here because they are set in separate initializer assignments
|
datatype.isNumericOrBool -> VarAllocation(address, datatype, size) // numerical variables in zeropage never have an initial value here because they are set in separate initializer assignments
|
||||||
DataType.STR -> VarAllocation(address, datatype, size)
|
datatype.isString -> VarAllocation(address, datatype, size)
|
||||||
in ArrayDatatypes -> VarAllocation(address, datatype, size)
|
datatype.isArray -> VarAllocation(address, datatype, size)
|
||||||
else -> throw AssemblyError("invalid dt")
|
else -> throw AssemblyError("invalid dt")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -150,14 +150,13 @@ class GoldenRam(options: CompilationOptions, val region: UIntRange): MemoryAlloc
|
|||||||
errors: IErrorReporter): Result<VarAllocation, MemAllocationError> {
|
errors: IErrorReporter): Result<VarAllocation, MemAllocationError> {
|
||||||
|
|
||||||
val size: Int =
|
val size: Int =
|
||||||
when (datatype) {
|
when {
|
||||||
in IntegerDatatypesWithBoolean -> options.compTarget.memorySize(datatype)
|
datatype.isIntegerOrBool -> options.compTarget.memorySize(datatype, null)
|
||||||
DataType.STR, in ArrayDatatypes -> {
|
datatype.isString -> numElements!!
|
||||||
options.compTarget.memorySize(datatype, numElements!!)
|
datatype.isArray -> options.compTarget.memorySize(datatype, numElements!!)
|
||||||
}
|
datatype.isFloat -> {
|
||||||
DataType.FLOAT -> {
|
|
||||||
if (options.floats) {
|
if (options.floats) {
|
||||||
options.compTarget.memorySize(DataType.FLOAT)
|
options.compTarget.memorySize(DataType.forDt(BaseDataType.FLOAT), null)
|
||||||
} else return Err(MemAllocationError("floating point option not enabled"))
|
} else return Err(MemAllocationError("floating point option not enabled"))
|
||||||
}
|
}
|
||||||
else -> throw MemAllocationError("weird dt")
|
else -> throw MemAllocationError("weird dt")
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
package prog8.code.core
|
package prog8.code.core
|
||||||
|
|
||||||
val AssociativeOperators = setOf("+", "*", "&", "|", "^", "==", "!=", "and", "or", "xor")
|
val AssociativeOperators = arrayOf("+", "*", "&", "|", "^", "==", "!=", "xor") // note: and,or are not associative because of Shortcircuit/McCarthy evaluation
|
||||||
val ComparisonOperators = setOf("==", "!=", "<", ">", "<=", ">=")
|
val ComparisonOperators = arrayOf("==", "!=", "<", ">", "<=", ">=")
|
||||||
val LogicalOperators = setOf("and", "or", "xor", "not", "in")
|
val LogicalOperators = arrayOf("and", "or", "xor", "not", "in")
|
||||||
val BitwiseOperators = setOf("&", "|", "^", "~")
|
val BitwiseOperators = arrayOf("&", "|", "^", "~")
|
||||||
val PrefixOperators = setOf("+", "-", "~", "not")
|
val PrefixOperators = arrayOf("+", "-", "~", "not")
|
||||||
|
|
||||||
fun invertedComparisonOperator(operator: String) =
|
fun invertedComparisonOperator(operator: String) =
|
||||||
when (operator) {
|
when (operator) {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
package prog8.code.core
|
package prog8.code.core
|
||||||
|
|
||||||
import prog8.code.core.SourceCode.Companion.LIBRARYFILEPREFIX
|
import prog8.code.source.SourceCode
|
||||||
import java.nio.file.InvalidPathException
|
import java.nio.file.InvalidPathException
|
||||||
import kotlin.io.path.Path
|
import kotlin.io.path.Path
|
||||||
import kotlin.io.path.absolute
|
import kotlin.io.path.absolute
|
||||||
@ -10,12 +10,12 @@ data class Position(val file: String, val line: Int, val startCol: Int, val endC
|
|||||||
fun toClickableStr(): String {
|
fun toClickableStr(): String {
|
||||||
if(this===DUMMY)
|
if(this===DUMMY)
|
||||||
return ""
|
return ""
|
||||||
if(file.startsWith(LIBRARYFILEPREFIX))
|
if(SourceCode.isLibraryResource(file))
|
||||||
return "$file:$line:$startCol:"
|
return "$file:$line:$startCol:"
|
||||||
return try {
|
return try {
|
||||||
val path = Path(file).absolute().normalize().toString()
|
val path = Path(file).absolute().normalize().toString()
|
||||||
"file://$path:$line:$startCol:"
|
"file://$path:$line:$startCol:"
|
||||||
} catch(x: InvalidPathException) {
|
} catch(_: InvalidPathException) {
|
||||||
// this can occur on Windows when the source origin contains "invalid" characters such as ':'
|
// this can occur on Windows when the source origin contains "invalid" characters such as ':'
|
||||||
"file://$file:$line:$startCol:"
|
"file://$file:$line:$startCol:"
|
||||||
}
|
}
|
||||||
|
@ -1,13 +1,18 @@
|
|||||||
package prog8.code.optimize
|
package prog8.code.optimize
|
||||||
|
|
||||||
|
import prog8.code.StExtSub
|
||||||
|
import prog8.code.SymbolTable
|
||||||
import prog8.code.ast.*
|
import prog8.code.ast.*
|
||||||
import prog8.code.core.*
|
import prog8.code.core.*
|
||||||
|
|
||||||
|
|
||||||
fun optimizeIntermediateAst(program: PtProgram, options: CompilationOptions, errors: IErrorReporter) {
|
fun optimizeIntermediateAst(program: PtProgram, options: CompilationOptions, st: SymbolTable, errors: IErrorReporter) {
|
||||||
if (!options.optimize)
|
if (!options.optimize)
|
||||||
return
|
return
|
||||||
while(errors.noErrors() && optimizeCommonSubExpressions(program, errors)>0) {
|
while (errors.noErrors() &&
|
||||||
|
(optimizeBitTest(program, options)
|
||||||
|
+ optimizeAssignTargets(program, st)) > 0
|
||||||
|
) {
|
||||||
// keep rolling
|
// keep rolling
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -22,109 +27,154 @@ private fun walkAst(root: PtNode, act: (node: PtNode, depth: Int) -> Boolean) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
private var tempVarCounter = 0
|
private fun optimizeAssignTargets(program: PtProgram, st: SymbolTable): Int {
|
||||||
|
var changes = 0
|
||||||
|
walkAst(program) { node: PtNode, depth: Int ->
|
||||||
|
if(node is PtAssignment) {
|
||||||
|
val value = node.value
|
||||||
|
val functionName = when(value) {
|
||||||
|
is PtBuiltinFunctionCall -> value.name
|
||||||
|
is PtFunctionCall -> value.name
|
||||||
|
else -> null
|
||||||
|
}
|
||||||
|
if(functionName!=null) {
|
||||||
|
val stNode = st.lookup(functionName)
|
||||||
|
if (stNode is StExtSub) {
|
||||||
|
require(node.children.size==stNode.returns.size+1) {
|
||||||
|
"number of targets must match return values"
|
||||||
|
}
|
||||||
|
node.children.zip(stNode.returns).withIndex().forEach { (index, xx) ->
|
||||||
|
val target = xx.first as PtAssignTarget
|
||||||
|
val returnedRegister = xx.second.register.registerOrPair
|
||||||
|
if(returnedRegister!=null && !target.void && target.identifier!=null) {
|
||||||
|
if(isSame(target.identifier!!, xx.second.type, returnedRegister)) {
|
||||||
|
// output register is already identical to target register, so it can become void
|
||||||
|
val voidTarget = PtAssignTarget(true, target.position)
|
||||||
|
node.children[index] = voidTarget
|
||||||
|
voidTarget.parent = node
|
||||||
|
changes++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(node.children.dropLast(1).all { (it as PtAssignTarget).void }) {
|
||||||
|
// all targets are now void, the whole assignment can be discarded and replaced by just a (void) call to the subroutine
|
||||||
|
val index = node.parent.children.indexOf(node)
|
||||||
|
val voidCall = PtFunctionCall(functionName, true, DataType.forDt(BaseDataType.UNDEFINED), value.position)
|
||||||
|
value.children.forEach { voidCall.add(it) }
|
||||||
|
node.parent.children[index] = voidCall
|
||||||
|
voidCall.parent = node.parent
|
||||||
|
changes++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
true
|
||||||
|
}
|
||||||
|
return changes
|
||||||
|
}
|
||||||
|
|
||||||
private fun optimizeCommonSubExpressions(program: PtProgram, errors: IErrorReporter): Int {
|
|
||||||
|
|
||||||
fun extractableSubExpr(expr: PtExpression): Boolean {
|
private fun optimizeBitTest(program: PtProgram, options: CompilationOptions): Int {
|
||||||
val result = if(expr is PtBinaryExpression)
|
if(options.compTarget.machine.cpu == CpuType.VIRTUAL)
|
||||||
expr.type !in ByteDatatypes ||
|
return 0 // the special bittest optimization is not yet valid for the IR
|
||||||
!expr.left.isSimple() ||
|
|
||||||
!expr.right.isSimple() ||
|
fun makeBittestCall(condition: PtBinaryExpression, and: PtBinaryExpression, variable: PtIdentifier, bitmask: Int): PtBuiltinFunctionCall {
|
||||||
(expr.operator !in LogicalOperators && expr.operator !in BitwiseOperators)
|
require(bitmask==128 || bitmask==64)
|
||||||
else if (expr is PtArrayIndexer && expr.type !in ByteDatatypes)
|
val setOrNot = if(condition.operator=="!=") "set" else "notset"
|
||||||
true
|
val bittestCall = PtBuiltinFunctionCall("prog8_ifelse_bittest_$setOrNot", false, true, DataType.forDt(BaseDataType.BOOL), condition.position)
|
||||||
|
bittestCall.add(variable)
|
||||||
|
if(bitmask==128)
|
||||||
|
bittestCall.add(PtNumber(BaseDataType.UBYTE, 7.0, and.right.position))
|
||||||
else
|
else
|
||||||
!expr.isSimple()
|
bittestCall.add(PtNumber(BaseDataType.UBYTE, 6.0, and.right.position))
|
||||||
return result
|
return bittestCall
|
||||||
}
|
}
|
||||||
|
|
||||||
// for each Binaryexpression, recurse to find a common subexpression pair therein.
|
fun isAndByteConditionForBRK(condition: PtBinaryExpression?): Triple<PtBinaryExpression, PtIdentifier, Int>? {
|
||||||
val commons = mutableMapOf<PtBinaryExpression, Pair<PtExpression, PtExpression>>()
|
if(condition!=null && (condition.operator=="==" || condition.operator=="!=")) {
|
||||||
walkAst(program) { node: PtNode, depth: Int ->
|
if (condition.right.asConstInteger() == 0) {
|
||||||
if(node is PtBinaryExpression) {
|
val and = condition.left as? PtBinaryExpression
|
||||||
val subExpressions = mutableListOf<PtExpression>()
|
if (and != null && and.operator == "&" && and.type.isUnsignedByte) {
|
||||||
walkAst(node.left) { subNode: PtNode, subDepth: Int ->
|
val bitmask = and.right.asConstInteger()
|
||||||
if (subNode is PtExpression) {
|
if(bitmask==128 || bitmask==64) {
|
||||||
if(extractableSubExpr(subNode)) subExpressions.add(subNode)
|
val variable = and.left as? PtIdentifier
|
||||||
true
|
if (variable != null && variable.type.isByte) {
|
||||||
} else false
|
return Triple(and, variable, bitmask)
|
||||||
}
|
}
|
||||||
walkAst(node.right) { subNode: PtNode, subDepth: Int ->
|
val typecast = and.left as? PtTypeCast
|
||||||
if (subNode is PtExpression) {
|
if (typecast != null && typecast.type.isUnsignedByte) {
|
||||||
if(extractableSubExpr(subNode)) subExpressions.add(subNode)
|
val castedVariable = typecast.value as? PtIdentifier
|
||||||
true
|
if(castedVariable!=null && castedVariable.type.isByte)
|
||||||
} else false
|
return Triple(and, castedVariable, bitmask)
|
||||||
}
|
}
|
||||||
|
|
||||||
outer@for (first in subExpressions) {
|
|
||||||
for (second in subExpressions) {
|
|
||||||
if (first!==second && first isSameAs second) {
|
|
||||||
commons[node] = first to second
|
|
||||||
break@outer // do only 1 replacement at a time per binaryexpression
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
false
|
|
||||||
} else true
|
|
||||||
}
|
|
||||||
|
|
||||||
// replace common subexpressions by a temp variable that is assigned only once.
|
|
||||||
// TODO: check for commonalities across multiple separate expressions in the current scope, not only inside a single line
|
|
||||||
commons.forEach { binexpr, (occurrence1, occurrence2) ->
|
|
||||||
val (stmtContainer, stmt) = findContainingStatements(binexpr)
|
|
||||||
val occurrence1idx = occurrence1.parent.children.indexOf(occurrence1)
|
|
||||||
val occurrence2idx = occurrence2.parent.children.indexOf(occurrence2)
|
|
||||||
val containerScopedName = findScopeName(stmtContainer)
|
|
||||||
tempVarCounter++
|
|
||||||
val tempvarName = "prog8_subexprvar_$tempVarCounter"
|
|
||||||
// TODO: some tempvars could be reused, if they are from different lines
|
|
||||||
|
|
||||||
val datatype = occurrence1.type
|
|
||||||
val singleReplacement1 = PtIdentifier("$containerScopedName.$tempvarName", datatype, occurrence1.position)
|
|
||||||
val singleReplacement2 = PtIdentifier("$containerScopedName.$tempvarName", datatype, occurrence2.position)
|
|
||||||
occurrence1.parent.children[occurrence1idx] = singleReplacement1
|
|
||||||
singleReplacement1.parent = occurrence1.parent
|
|
||||||
occurrence2.parent.children[occurrence2idx] = singleReplacement2
|
|
||||||
singleReplacement2.parent = occurrence2.parent
|
|
||||||
|
|
||||||
val tempassign = PtAssignment(binexpr.position).also { assign ->
|
|
||||||
assign.add(PtAssignTarget(binexpr.position).also { tgt->
|
|
||||||
tgt.add(PtIdentifier("$containerScopedName.$tempvarName", datatype, binexpr.position))
|
|
||||||
})
|
|
||||||
assign.add(occurrence1)
|
|
||||||
occurrence1.parent = assign
|
|
||||||
}
|
}
|
||||||
stmtContainer.children.add(stmtContainer.children.indexOf(stmt), tempassign)
|
return null
|
||||||
tempassign.parent = stmtContainer
|
|
||||||
|
|
||||||
val tempvar = PtVariable(tempvarName, datatype, ZeropageWish.NOT_IN_ZEROPAGE, null, null, binexpr.position)
|
|
||||||
stmtContainer.add(0, tempvar)
|
|
||||||
tempvar.parent = stmtContainer
|
|
||||||
|
|
||||||
// errors.info("common subexpressions replaced by a tempvar, maybe simplify the expression manually", binexpr.position)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return commons.size
|
var changes = 0
|
||||||
}
|
var recurse = true
|
||||||
|
walkAst(program) { node: PtNode, depth: Int ->
|
||||||
|
if(node is PtIfElse) {
|
||||||
internal fun findScopeName(node: PtNode): String {
|
val condition = node.condition as? PtBinaryExpression
|
||||||
var parent=node
|
val check = isAndByteConditionForBRK(condition)
|
||||||
while(parent !is PtNamedNode)
|
if(check!=null) {
|
||||||
parent = parent.parent
|
val (and, variable, bitmask) = check
|
||||||
return parent.scopedName
|
val bittestCall = makeBittestCall(condition!!, and, variable, bitmask)
|
||||||
}
|
val ifElse = PtIfElse(node.position)
|
||||||
|
ifElse.add(bittestCall)
|
||||||
|
ifElse.add(node.ifScope)
|
||||||
internal fun findContainingStatements(node: PtNode): Pair<PtNode, PtNode> { // returns (parentstatementcontainer, childstatement)
|
if (node.hasElse())
|
||||||
var parent = node.parent
|
ifElse.add(node.elseScope)
|
||||||
var child = node
|
val index = node.parent.children.indexOf(node)
|
||||||
while(true) {
|
node.parent.children[index] = ifElse
|
||||||
if(parent is IPtStatementContainer) {
|
ifElse.parent = node.parent
|
||||||
return parent to child
|
changes++
|
||||||
|
recurse = false
|
||||||
|
}
|
||||||
}
|
}
|
||||||
child=parent
|
if (node is PtIfExpression) {
|
||||||
parent=parent.parent
|
val condition = node.condition as? PtBinaryExpression
|
||||||
|
val check = isAndByteConditionForBRK(condition)
|
||||||
|
if(check!=null) {
|
||||||
|
val (and, variable, bitmask) = check
|
||||||
|
val bittestCall = makeBittestCall(condition!!, and, variable, bitmask)
|
||||||
|
node.children[0] = bittestCall
|
||||||
|
bittestCall.parent = node
|
||||||
|
changes++
|
||||||
|
recurse = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
recurse
|
||||||
}
|
}
|
||||||
|
return changes
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
internal fun isSame(identifier: PtIdentifier, type: DataType, returnedRegister: RegisterOrPair): Boolean {
|
||||||
|
if(returnedRegister in Cx16VirtualRegisters) {
|
||||||
|
val regname = returnedRegister.name.lowercase()
|
||||||
|
val identifierRegName = identifier.name.substringAfterLast('.')
|
||||||
|
/*
|
||||||
|
cx16.r? UWORD
|
||||||
|
cx16.r?s WORD
|
||||||
|
cx16.r?L UBYTE
|
||||||
|
cx16.r?H UBYTE
|
||||||
|
cx16.r?sL BYTE
|
||||||
|
cx16.r?sH BYTE
|
||||||
|
*/
|
||||||
|
if(identifier.type.isByte && type.isByte) {
|
||||||
|
if(identifier.name.startsWith("cx16.$regname") && identifierRegName.startsWith(regname)) {
|
||||||
|
return identifierRegName.substring(2) in arrayOf("", "L", "sL") // note: not the -H (msb) variants!
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if(identifier.type.isWord && type.isWord) {
|
||||||
|
if(identifier.name.startsWith("cx16.$regname") && identifierRegName.startsWith(regname)) {
|
||||||
|
return identifierRegName.substring(2) in arrayOf("", "s")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false // there are no identifiers directly corresponding to cpu registers
|
||||||
}
|
}
|
||||||
|
74
codeCore/src/prog8/code/source/ImportFileSystem.kt
Normal file
74
codeCore/src/prog8/code/source/ImportFileSystem.kt
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
package prog8.code.source
|
||||||
|
|
||||||
|
import prog8.code.core.Position
|
||||||
|
import java.nio.file.Path
|
||||||
|
import java.util.TreeMap
|
||||||
|
import kotlin.io.path.Path
|
||||||
|
import kotlin.io.path.absolute
|
||||||
|
|
||||||
|
|
||||||
|
// Resource caching "filesystem".
|
||||||
|
// Note that it leaves the decision to load a resource or an actual disk file to the caller.
|
||||||
|
|
||||||
|
object ImportFileSystem {
|
||||||
|
|
||||||
|
fun expandTilde(path: String): String = if (path.startsWith("~")) {
|
||||||
|
val userHome = System.getProperty("user.home")
|
||||||
|
userHome + path.drop(1)
|
||||||
|
} else {
|
||||||
|
path
|
||||||
|
}
|
||||||
|
|
||||||
|
fun expandTilde(path: Path): Path = Path(expandTilde(path.toString()))
|
||||||
|
|
||||||
|
fun getFile(path: Path): SourceCode {
|
||||||
|
val normalized = path.absolute().normalize()
|
||||||
|
val cached = cache[normalized.toString()]
|
||||||
|
if (cached != null)
|
||||||
|
return cached
|
||||||
|
val file = SourceCode.File(normalized)
|
||||||
|
cache[normalized.toString()] = file
|
||||||
|
return file
|
||||||
|
}
|
||||||
|
|
||||||
|
fun getResource(name: String): SourceCode {
|
||||||
|
val cached = cache[name]
|
||||||
|
if (cached != null) return cached
|
||||||
|
val resource = SourceCode.Resource(name)
|
||||||
|
cache[name] = resource
|
||||||
|
return resource
|
||||||
|
}
|
||||||
|
|
||||||
|
fun retrieveSourceLine(position: Position): String {
|
||||||
|
if(SourceCode.isLibraryResource(position.file)) {
|
||||||
|
val cached = cache[SourceCode.withoutPrefix(position.file)]
|
||||||
|
if(cached != null)
|
||||||
|
return getLine(cached, position.line)
|
||||||
|
}
|
||||||
|
val cached = cache[position.file]
|
||||||
|
if(cached != null)
|
||||||
|
return getLine(cached, position.line)
|
||||||
|
val path = Path(position.file).absolute().normalize()
|
||||||
|
val cached2 = cache[path.toString()]
|
||||||
|
if(cached2 != null)
|
||||||
|
return getLine(cached2, position.line)
|
||||||
|
throw NoSuchElementException("cannot get source line $position, with path $path")
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun getLine(code: SourceCode, lineIndex: Int): String {
|
||||||
|
var spans = lineSpanCache[code]
|
||||||
|
if(spans==null) {
|
||||||
|
val lineSpans = Regex("^", RegexOption.MULTILINE).findAll(code.text).map { it.range.first }
|
||||||
|
val ends = lineSpans.drop(1) + code.text.length
|
||||||
|
spans = lineSpans.zip(ends).map { (start, end) -> LineSpan(start, end) }.toList().toTypedArray()
|
||||||
|
lineSpanCache[code] = spans
|
||||||
|
}
|
||||||
|
val span = spans[lineIndex - 1]
|
||||||
|
return code.text.substring(span.start, span.end).trim()
|
||||||
|
}
|
||||||
|
|
||||||
|
private class LineSpan(val start: Int, val end: Int)
|
||||||
|
|
||||||
|
private val cache = TreeMap<String, SourceCode>(String.CASE_INSENSITIVE_ORDER)
|
||||||
|
private val lineSpanCache = mutableMapOf<SourceCode, Array<LineSpan>>()
|
||||||
|
}
|
@ -1,16 +1,12 @@
|
|||||||
package prog8.code.core
|
package prog8.code.source
|
||||||
|
|
||||||
import java.io.File
|
|
||||||
import java.io.IOException
|
import java.io.IOException
|
||||||
import java.nio.file.Path
|
import java.nio.file.Path
|
||||||
import java.text.Normalizer
|
import java.text.Normalizer
|
||||||
import kotlin.io.path.Path
|
import kotlin.io.path.Path
|
||||||
|
import kotlin.io.path.absolute
|
||||||
import kotlin.io.path.readText
|
import kotlin.io.path.readText
|
||||||
|
|
||||||
|
|
||||||
const val internedStringsModuleName = "prog8_interned_strings"
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Encapsulates - and ties together - actual source code (=text) and its [origin].
|
* Encapsulates - and ties together - actual source code (=text) and its [origin].
|
||||||
*/
|
*/
|
||||||
@ -55,14 +51,21 @@ sealed class SourceCode {
|
|||||||
/**
|
/**
|
||||||
* filename prefix to designate library files that will be retreived from internal resources rather than disk
|
* filename prefix to designate library files that will be retreived from internal resources rather than disk
|
||||||
*/
|
*/
|
||||||
const val LIBRARYFILEPREFIX = "library:"
|
private const val LIBRARYFILEPREFIX = "library:"
|
||||||
const val STRINGSOURCEPREFIX = "string:"
|
private const val STRINGSOURCEPREFIX = "string:"
|
||||||
val curdir: Path = Path(".").toAbsolutePath()
|
val curdir: Path = Path(".").absolute()
|
||||||
fun relative(path: Path): Path = curdir.relativize(path.toAbsolutePath())
|
fun relative(path: Path): Path = curdir.relativize(path.absolute())
|
||||||
fun isRegularFilesystemPath(pathString: String) =
|
fun isRegularFilesystemPath(pathString: String) = !isLibraryResource(pathString) && !isStringResource(pathString)
|
||||||
!(pathString.startsWith(LIBRARYFILEPREFIX) || pathString.startsWith(STRINGSOURCEPREFIX))
|
|
||||||
|
|
||||||
fun isLibraryResource(path: String) = path.startsWith(LIBRARYFILEPREFIX)
|
fun isLibraryResource(path: String) = path.startsWith(LIBRARYFILEPREFIX)
|
||||||
|
fun isStringResource(path: String) = path.startsWith(STRINGSOURCEPREFIX)
|
||||||
|
fun withoutPrefix(path: String): String {
|
||||||
|
return if(isLibraryResource(path))
|
||||||
|
path.removePrefix(LIBRARYFILEPREFIX)
|
||||||
|
else if(isStringResource(path))
|
||||||
|
path.removePrefix(STRINGSOURCEPREFIX)
|
||||||
|
else
|
||||||
|
path
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -80,12 +83,13 @@ sealed class SourceCode {
|
|||||||
/**
|
/**
|
||||||
* Get [SourceCode] from the file represented by the specified Path.
|
* Get [SourceCode] from the file represented by the specified Path.
|
||||||
* This immediately reads the file fully into memory.
|
* This immediately reads the file fully into memory.
|
||||||
|
* You can only get an instance of this via the ImportFileSystem object.
|
||||||
*
|
*
|
||||||
* [origin] will be the given path in absolute and normalized form.
|
* [origin] will be the given path in absolute and normalized form.
|
||||||
* @throws NoSuchFileException if the file does not exist
|
* @throws NoSuchFileException if the file does not exist
|
||||||
* @throws FileSystemException if the file cannot be read
|
* @throws FileSystemException if the file cannot be read
|
||||||
*/
|
*/
|
||||||
class File(path: Path): SourceCode() {
|
internal class File(path: Path): SourceCode() {
|
||||||
override val text: String
|
override val text: String
|
||||||
override val origin: String
|
override val origin: String
|
||||||
override val name: String
|
override val name: String
|
||||||
@ -109,8 +113,9 @@ sealed class SourceCode {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* [origin]: `library:/x/y/z.p8` for a given `pathString` of "x/y/z.p8"
|
* [origin]: `library:/x/y/z.p8` for a given `pathString` of "x/y/z.p8"
|
||||||
|
* You can only get an instance of this via the ImportFileSystem object.
|
||||||
*/
|
*/
|
||||||
class Resource(pathString: String): SourceCode() {
|
internal class Resource(pathString: String): SourceCode() {
|
||||||
private val normalized = "/" + Path(pathString).normalize().toMutableList().joinToString("/")
|
private val normalized = "/" + Path(pathString).normalize().toMutableList().joinToString("/")
|
||||||
|
|
||||||
override val isFromResources = true
|
override val isFromResources = true
|
||||||
@ -124,7 +129,7 @@ sealed class SourceCode {
|
|||||||
if (rscURL == null) {
|
if (rscURL == null) {
|
||||||
val rscRoot = object {}.javaClass.getResource("/")
|
val rscRoot = object {}.javaClass.getResource("/")
|
||||||
throw NoSuchFileException(
|
throw NoSuchFileException(
|
||||||
File(normalized),
|
java.io.File(normalized),
|
||||||
reason = "looked in resources rooted at $rscRoot"
|
reason = "looked in resources rooted at $rscRoot"
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -144,34 +149,4 @@ sealed class SourceCode {
|
|||||||
override val origin: String = name
|
override val origin: String = name
|
||||||
override val text: String = "<generated code node, no text representation>"
|
override val text: String = "<generated code node, no text representation>"
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
object SourceLineCache {
|
|
||||||
private val cache = mutableMapOf<String, List<String>>()
|
|
||||||
|
|
||||||
private fun getCachedFile(file: String): List<String> {
|
|
||||||
val existing = cache[file]
|
|
||||||
if(existing!=null)
|
|
||||||
return existing
|
|
||||||
if (SourceCode.isRegularFilesystemPath(file)) {
|
|
||||||
val source = SourceCode.File(Path(file))
|
|
||||||
cache[file] = source.text.split('\n', '\r').map { it.trim() }
|
|
||||||
return cache.getValue(file)
|
|
||||||
} else if(file.startsWith(SourceCode.LIBRARYFILEPREFIX)) {
|
|
||||||
val source = SourceCode.Resource(file.drop(SourceCode.LIBRARYFILEPREFIX.length))
|
|
||||||
cache[file] = source.text.split('\n', '\r').map { it.trim()}
|
|
||||||
return cache.getValue(file)
|
|
||||||
}
|
|
||||||
return emptyList()
|
|
||||||
}
|
|
||||||
|
|
||||||
fun retrieveLine(position: Position): String? {
|
|
||||||
if (position.line>0) {
|
|
||||||
val lines = getCachedFile(position.file)
|
|
||||||
if(lines.isNotEmpty())
|
|
||||||
return lines[position.line-1]
|
|
||||||
}
|
|
||||||
return null
|
|
||||||
}
|
|
||||||
}
|
}
|
@ -13,19 +13,28 @@ class AtariTarget: ICompilationTarget, IStringEncoding by Encoder, IMemSizer {
|
|||||||
const val NAME = "atari"
|
const val NAME = "atari"
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun memorySize(dt: DataType): Int {
|
override fun memorySize(dt: DataType, numElements: Int?): Int {
|
||||||
return when(dt) {
|
if(dt.isArray) {
|
||||||
in ByteDatatypesWithBoolean -> 1
|
if(numElements==null) return 2 // treat it as a pointer size
|
||||||
in WordDatatypes, in PassByReferenceDatatypes -> 2
|
return when(dt.sub) {
|
||||||
DataType.FLOAT -> machine.FLOAT_MEM_SIZE
|
BaseDataType.BOOL, BaseDataType.UBYTE, BaseDataType.BYTE -> numElements
|
||||||
else -> throw IllegalArgumentException("invalid datatype")
|
BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.STR -> numElements * 2
|
||||||
|
BaseDataType.FLOAT-> numElements * machine.FLOAT_MEM_SIZE
|
||||||
|
BaseDataType.UNDEFINED -> throw IllegalArgumentException("undefined has no memory size")
|
||||||
|
else -> throw IllegalArgumentException("invalid sub type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (dt.isString) {
|
||||||
|
if(numElements!=null) return numElements // treat it as the size of the given string with the length
|
||||||
|
else return 2 // treat it as the size to store a string pointer
|
||||||
|
}
|
||||||
|
|
||||||
|
return when {
|
||||||
|
dt.isByteOrBool -> 1 * (numElements ?: 1)
|
||||||
|
dt.isFloat -> machine.FLOAT_MEM_SIZE * (numElements ?: 1)
|
||||||
|
dt.isLong -> throw IllegalArgumentException("long can not yet be put into memory")
|
||||||
|
dt.isUndefined -> throw IllegalArgumentException("undefined has no memory size")
|
||||||
|
else -> 2 * (numElements ?: 1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun memorySize(arrayDt: DataType, numElements: Int) =
|
|
||||||
if(arrayDt==DataType.UWORD)
|
|
||||||
numElements // pointer to bytes.
|
|
||||||
else
|
|
||||||
memorySize(ArrayToElementTypes.getValue(arrayDt)) * numElements
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -19,3 +19,25 @@ class C64Target: ICompilationTarget, IStringEncoding by Encoder, IMemSizer by Cb
|
|||||||
fun viceMonListName(baseFilename: String) = "$baseFilename.vice-mon-list"
|
fun viceMonListName(baseFilename: String) = "$baseFilename.vice-mon-list"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
val CompilationTargets = listOf(
|
||||||
|
C64Target.NAME,
|
||||||
|
C128Target.NAME,
|
||||||
|
Cx16Target.NAME,
|
||||||
|
PETTarget.NAME,
|
||||||
|
AtariTarget.NAME,
|
||||||
|
Neo6502Target.NAME,
|
||||||
|
VMTarget.NAME
|
||||||
|
)
|
||||||
|
|
||||||
|
fun getCompilationTargetByName(name: String) = when(name.lowercase()) {
|
||||||
|
C64Target.NAME -> C64Target()
|
||||||
|
C128Target.NAME -> C128Target()
|
||||||
|
Cx16Target.NAME -> Cx16Target()
|
||||||
|
PETTarget.NAME -> PETTarget()
|
||||||
|
AtariTarget.NAME -> AtariTarget()
|
||||||
|
VMTarget.NAME -> VMTarget()
|
||||||
|
Neo6502Target.NAME -> Neo6502Target()
|
||||||
|
else -> throw IllegalArgumentException("invalid compilation target")
|
||||||
|
}
|
||||||
|
@ -4,9 +4,7 @@ import com.github.michaelbull.result.fold
|
|||||||
import prog8.code.core.Encoding
|
import prog8.code.core.Encoding
|
||||||
import prog8.code.core.IStringEncoding
|
import prog8.code.core.IStringEncoding
|
||||||
import prog8.code.core.InternalCompilerException
|
import prog8.code.core.InternalCompilerException
|
||||||
import prog8.code.target.cbm.AtasciiEncoding
|
import prog8.code.target.encodings.*
|
||||||
import prog8.code.target.cbm.IsoEncoding
|
|
||||||
import prog8.code.target.cbm.PetsciiEncoding
|
|
||||||
|
|
||||||
|
|
||||||
object Encoder: IStringEncoding {
|
object Encoder: IStringEncoding {
|
||||||
@ -18,6 +16,10 @@ object Encoder: IStringEncoding {
|
|||||||
Encoding.SCREENCODES -> PetsciiEncoding.encodeScreencode(str, true)
|
Encoding.SCREENCODES -> PetsciiEncoding.encodeScreencode(str, true)
|
||||||
Encoding.ISO -> IsoEncoding.encode(str)
|
Encoding.ISO -> IsoEncoding.encode(str)
|
||||||
Encoding.ATASCII -> AtasciiEncoding.encode(str)
|
Encoding.ATASCII -> AtasciiEncoding.encode(str)
|
||||||
|
Encoding.ISO5 -> IsoCyrillicEncoding.encode(str)
|
||||||
|
Encoding.ISO16 -> IsoEasternEncoding.encode(str)
|
||||||
|
Encoding.CP437 -> Cp437Encoding.encode(str)
|
||||||
|
Encoding.KATAKANA -> KatakanaEncoding.encode(str)
|
||||||
else -> throw InternalCompilerException("unsupported encoding $encoding")
|
else -> throw InternalCompilerException("unsupported encoding $encoding")
|
||||||
}
|
}
|
||||||
return coded.fold(
|
return coded.fold(
|
||||||
@ -31,6 +33,10 @@ object Encoder: IStringEncoding {
|
|||||||
Encoding.SCREENCODES -> PetsciiEncoding.decodeScreencode(bytes, true)
|
Encoding.SCREENCODES -> PetsciiEncoding.decodeScreencode(bytes, true)
|
||||||
Encoding.ISO -> IsoEncoding.decode(bytes)
|
Encoding.ISO -> IsoEncoding.decode(bytes)
|
||||||
Encoding.ATASCII -> AtasciiEncoding.decode(bytes)
|
Encoding.ATASCII -> AtasciiEncoding.decode(bytes)
|
||||||
|
Encoding.ISO5 -> IsoCyrillicEncoding.decode(bytes)
|
||||||
|
Encoding.ISO16 -> IsoEasternEncoding.decode(bytes)
|
||||||
|
Encoding.CP437 -> Cp437Encoding.decode(bytes)
|
||||||
|
Encoding.KATAKANA -> KatakanaEncoding.decode(bytes)
|
||||||
else -> throw InternalCompilerException("unsupported encoding $encoding")
|
else -> throw InternalCompilerException("unsupported encoding $encoding")
|
||||||
}
|
}
|
||||||
return decoded.fold(
|
return decoded.fold(
|
||||||
|
40
codeCore/src/prog8/code/target/Neo6502Target.kt
Normal file
40
codeCore/src/prog8/code/target/Neo6502Target.kt
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
package prog8.code.target
|
||||||
|
|
||||||
|
import prog8.code.core.*
|
||||||
|
import prog8.code.target.neo6502.Neo6502MachineDefinition
|
||||||
|
|
||||||
|
|
||||||
|
class Neo6502Target: ICompilationTarget, IStringEncoding by Encoder, IMemSizer {
|
||||||
|
override val name = NAME
|
||||||
|
override val machine = Neo6502MachineDefinition()
|
||||||
|
override val defaultEncoding = Encoding.ISO
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
const val NAME = "neo"
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun memorySize(dt: DataType, numElements: Int?): Int {
|
||||||
|
if(dt.isArray) {
|
||||||
|
if(numElements==null) return 2 // treat it as a pointer size
|
||||||
|
return when(dt.sub) {
|
||||||
|
BaseDataType.BOOL, BaseDataType.UBYTE, BaseDataType.BYTE -> numElements
|
||||||
|
BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.STR -> numElements * 2
|
||||||
|
BaseDataType.FLOAT-> numElements * machine.FLOAT_MEM_SIZE
|
||||||
|
BaseDataType.UNDEFINED -> throw IllegalArgumentException("undefined has no memory size")
|
||||||
|
else -> throw IllegalArgumentException("invalid sub type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (dt.isString) {
|
||||||
|
if(numElements!=null) return numElements // treat it as the size of the given string with the length
|
||||||
|
else return 2 // treat it as the size to store a string pointer
|
||||||
|
}
|
||||||
|
|
||||||
|
return when {
|
||||||
|
dt.isByteOrBool -> 1 * (numElements ?: 1)
|
||||||
|
dt.isFloat -> machine.FLOAT_MEM_SIZE * (numElements ?: 1)
|
||||||
|
dt.isLong -> throw IllegalArgumentException("long can not yet be put into memory")
|
||||||
|
dt.isUndefined -> throw IllegalArgumentException("undefined has no memory size")
|
||||||
|
else -> 2 * (numElements ?: 1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -12,18 +12,28 @@ class VMTarget: ICompilationTarget, IStringEncoding by Encoder, IMemSizer {
|
|||||||
const val NAME = "virtual"
|
const val NAME = "virtual"
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun memorySize(dt: DataType): Int {
|
override fun memorySize(dt: DataType, numElements: Int?): Int {
|
||||||
return when(dt) {
|
if(dt.isArray) {
|
||||||
in ByteDatatypesWithBoolean -> 1
|
if(numElements==null) return 2 // treat it as a pointer size
|
||||||
in WordDatatypes, in PassByReferenceDatatypes -> 2
|
return when(dt.sub) {
|
||||||
DataType.FLOAT -> machine.FLOAT_MEM_SIZE
|
BaseDataType.BOOL, BaseDataType.UBYTE, BaseDataType.BYTE -> numElements
|
||||||
else -> throw IllegalArgumentException("invalid datatype")
|
BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.STR -> numElements * 2
|
||||||
|
BaseDataType.FLOAT-> numElements * machine.FLOAT_MEM_SIZE
|
||||||
|
BaseDataType.UNDEFINED -> throw IllegalArgumentException("undefined has no memory size")
|
||||||
|
else -> throw IllegalArgumentException("invalid sub type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (dt.isString) {
|
||||||
|
if(numElements!=null) return numElements // treat it as the size of the given string with the length
|
||||||
|
else return 2 // treat it as the size to store a string pointer
|
||||||
|
}
|
||||||
|
|
||||||
|
return when {
|
||||||
|
dt.isByteOrBool -> 1 * (numElements ?: 1)
|
||||||
|
dt.isFloat -> machine.FLOAT_MEM_SIZE * (numElements ?: 1)
|
||||||
|
dt.isLong -> throw IllegalArgumentException("long can not yet be put into memory")
|
||||||
|
dt.isUndefined -> throw IllegalArgumentException("undefined has no memory size")
|
||||||
|
else -> 2 * (numElements ?: 1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun memorySize(arrayDt: DataType, numElements: Int) =
|
|
||||||
if(arrayDt==DataType.UWORD)
|
|
||||||
numElements // pointer to bytes.
|
|
||||||
else
|
|
||||||
memorySize(ArrayToElementTypes.getValue(arrayDt)) * numElements
|
|
||||||
}
|
}
|
@ -11,7 +11,9 @@ class AtariMachineDefinition: IMachineDefinition {
|
|||||||
override val FLOAT_MAX_POSITIVE = 9.999999999e97
|
override val FLOAT_MAX_POSITIVE = 9.999999999e97
|
||||||
override val FLOAT_MAX_NEGATIVE = -9.999999999e97
|
override val FLOAT_MAX_NEGATIVE = -9.999999999e97
|
||||||
override val FLOAT_MEM_SIZE = 6
|
override val FLOAT_MEM_SIZE = 6
|
||||||
|
override val STARTUP_CODE_RESERVED_SIZE = 20u
|
||||||
override val PROGRAM_LOAD_ADDRESS = 0x2000u
|
override val PROGRAM_LOAD_ADDRESS = 0x2000u
|
||||||
|
override val PROGRAM_MEMTOP_ADDRESS = 0xffffu // TODO what's memtop?
|
||||||
|
|
||||||
override val BSSHIGHRAM_START = 0u // TODO
|
override val BSSHIGHRAM_START = 0u // TODO
|
||||||
override val BSSHIGHRAM_END = 0u // TODO
|
override val BSSHIGHRAM_END = 0u // TODO
|
||||||
@ -22,13 +24,8 @@ class AtariMachineDefinition: IMachineDefinition {
|
|||||||
override lateinit var golden: GoldenRam
|
override lateinit var golden: GoldenRam
|
||||||
|
|
||||||
override fun getFloatAsmBytes(num: Number) = TODO("atari float asm bytes from number")
|
override fun getFloatAsmBytes(num: Number) = TODO("atari float asm bytes from number")
|
||||||
|
override fun convertFloatToBytes(num: Double): List<UByte> = TODO("atari float to bytes")
|
||||||
override fun importLibs(compilerOptions: CompilationOptions, compilationTargetName: String): List<String> {
|
override fun convertBytesToFloat(bytes: List<UByte>): Double = TODO("atari bytes to float")
|
||||||
return if (compilerOptions.output == OutputType.XEX)
|
|
||||||
listOf("syslib")
|
|
||||||
else
|
|
||||||
emptyList()
|
|
||||||
}
|
|
||||||
|
|
||||||
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
||||||
val emulatorName: String
|
val emulatorName: String
|
||||||
|
@ -28,7 +28,7 @@ class AtariZeropage(options: CompilationOptions) : Zeropage(options) {
|
|||||||
ZeropageType.FULL -> {
|
ZeropageType.FULL -> {
|
||||||
// TODO all atari usable zero page locations, except the ones used by the system's IRQ routine
|
// TODO all atari usable zero page locations, except the ones used by the system's IRQ routine
|
||||||
free.addAll(0x00u..0xffu)
|
free.addAll(0x00u..0xffu)
|
||||||
// TODO atari free.removeAll(setOf(0xa0u, 0xa1u, 0xa2u, 0x91u, 0xc0u, 0xc5u, 0xcbu, 0xf5u, 0xf6u)) // these are updated by IRQ
|
// TODO atari free.removeAll(arrayOf(0xa0u, 0xa1u, 0xa2u, 0x91u, 0xc0u, 0xc5u, 0xcbu, 0xf5u, 0xf6u)) // these are updated by IRQ
|
||||||
}
|
}
|
||||||
ZeropageType.KERNALSAFE -> {
|
ZeropageType.KERNALSAFE -> {
|
||||||
free.addAll(0x80u..0xffu) // TODO
|
free.addAll(0x80u..0xffu) // TODO
|
||||||
|
@ -13,7 +13,9 @@ class C128MachineDefinition: IMachineDefinition {
|
|||||||
override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE
|
override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE
|
||||||
override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE
|
override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE
|
||||||
override val FLOAT_MEM_SIZE = Mflpt5.FLOAT_MEM_SIZE
|
override val FLOAT_MEM_SIZE = Mflpt5.FLOAT_MEM_SIZE
|
||||||
|
override val STARTUP_CODE_RESERVED_SIZE = 20u
|
||||||
override val PROGRAM_LOAD_ADDRESS = 0x1c01u
|
override val PROGRAM_LOAD_ADDRESS = 0x1c01u
|
||||||
|
override val PROGRAM_MEMTOP_ADDRESS = 0xc000u
|
||||||
|
|
||||||
override val BSSHIGHRAM_START = 0u // TODO
|
override val BSSHIGHRAM_START = 0u // TODO
|
||||||
override val BSSHIGHRAM_END = 0u // TODO
|
override val BSSHIGHRAM_END = 0u // TODO
|
||||||
@ -25,11 +27,15 @@ class C128MachineDefinition: IMachineDefinition {
|
|||||||
|
|
||||||
override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm()
|
override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm()
|
||||||
|
|
||||||
override fun importLibs(compilerOptions: CompilationOptions, compilationTargetName: String): List<String> {
|
override fun convertFloatToBytes(num: Double): List<UByte> {
|
||||||
return if (compilerOptions.launcher == CbmPrgLauncherType.BASIC || compilerOptions.output == OutputType.PRG)
|
val m5 = Mflpt5.fromNumber(num)
|
||||||
listOf("syslib")
|
return listOf(m5.b0, m5.b1, m5.b2, m5.b3, m5.b4)
|
||||||
else
|
}
|
||||||
emptyList()
|
|
||||||
|
override fun convertBytesToFloat(bytes: List<UByte>): Double {
|
||||||
|
require(bytes.size==5) { "need 5 bytes" }
|
||||||
|
val m5 = Mflpt5(bytes[0], bytes[1], bytes[2], bytes[3], bytes[4])
|
||||||
|
return m5.toDouble()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
||||||
|
@ -6,7 +6,7 @@ import prog8.code.core.Zeropage
|
|||||||
import prog8.code.core.ZeropageType
|
import prog8.code.core.ZeropageType
|
||||||
|
|
||||||
|
|
||||||
// reference: "Mapping the C128" zero page chapter.
|
// reference: "Mapping the C128" zeropage chapter.
|
||||||
|
|
||||||
class C128Zeropage(options: CompilationOptions) : Zeropage(options) {
|
class C128Zeropage(options: CompilationOptions) : Zeropage(options) {
|
||||||
|
|
||||||
@ -33,18 +33,18 @@ class C128Zeropage(options: CompilationOptions) : Zeropage(options) {
|
|||||||
ZeropageType.FULL -> {
|
ZeropageType.FULL -> {
|
||||||
// $00/$01 are data port IO registers, // $02-$09 are storage locations for JSRFAR and such
|
// $00/$01 are data port IO registers, // $02-$09 are storage locations for JSRFAR and such
|
||||||
free.addAll(0x0au..0xffu)
|
free.addAll(0x0au..0xffu)
|
||||||
free.removeAll(listOf(0x90u, 0x91u, 0xa0u, 0xa1u, 0xa2u, 0xc0u, 0xccu, 0xcdu, 0xd0u, 0xd1u, 0xd2u, 0xd3u, 0xd4u, 0xd5u, 0xf7u)) // these are updated/used by IRQ
|
free.removeAll(arrayOf(0x90u, 0x91u, 0xa0u, 0xa1u, 0xa2u, 0xc0u, 0xccu, 0xcdu, 0xd0u, 0xd1u, 0xd2u, 0xd3u, 0xd4u, 0xd5u, 0xf7u)) // these are updated/used by IRQ
|
||||||
}
|
}
|
||||||
ZeropageType.KERNALSAFE -> {
|
ZeropageType.KERNALSAFE -> {
|
||||||
free.addAll(0x0au..0x8fu) // BASIC variables
|
free.addAll(0x0au..0x8fu) // BASIC variables
|
||||||
free.addAll(listOf(0x92u, 0x96u, 0x9bu, 0x9cu, 0x9eu, 0x9fu, 0xa4u, 0xa7u, 0xa8u, 0xa9u, 0xaau, 0xabu,
|
free.addAll(arrayOf(0x92u, 0x96u, 0x9bu, 0x9cu, 0x9eu, 0x9fu, 0xa4u, 0xa7u, 0xa8u, 0xa9u, 0xaau, 0xabu,
|
||||||
0xb0u, 0xb1u, 0xb4u, 0xb5u, 0xb6u))
|
0xb0u, 0xb1u, 0xb4u, 0xb5u, 0xb6u))
|
||||||
}
|
}
|
||||||
ZeropageType.FLOATSAFE,
|
ZeropageType.FLOATSAFE,
|
||||||
ZeropageType.BASICSAFE -> {
|
ZeropageType.BASICSAFE -> {
|
||||||
free.addAll(listOf(0x0bu, 0x0cu, 0x0du, 0x0eu, 0x0fu, 0x10u, 0x11u, 0x12u, 0x16u, 0x17u, 0x18u, 0x19u, 0x1au))
|
free.addAll(arrayOf(0x0bu, 0x0cu, 0x0du, 0x0eu, 0x0fu, 0x10u, 0x11u, 0x12u, 0x16u, 0x17u, 0x18u, 0x19u, 0x1au))
|
||||||
free.addAll(0x1bu..0x23u)
|
free.addAll(0x1bu..0x23u)
|
||||||
free.addAll(listOf(0x3fu, 0x40u, 0x41u, 0x42u, 0x43u, 0x44u, 0x47u, 0x48u, 0x49u, 0x4au, 0x4bu, 0x4cu, 0x4fu,
|
free.addAll(arrayOf(0x3fu, 0x40u, 0x41u, 0x42u, 0x43u, 0x44u, 0x47u, 0x48u, 0x49u, 0x4au, 0x4bu, 0x4cu, 0x4fu,
|
||||||
0x55u, 0x56u, 0x57u, 0x58u,
|
0x55u, 0x56u, 0x57u, 0x58u,
|
||||||
0x74u, 0x75u, 0x78u, 0x80u, 0x83u, 0x87u, 0x88u, 0x89u, 0x8au, 0x8bu, 0x8cu, 0x8du, 0x8eu, 0x8fu,
|
0x74u, 0x75u, 0x78u, 0x80u, 0x83u, 0x87u, 0x88u, 0x89u, 0x8au, 0x8bu, 0x8cu, 0x8du, 0x8eu, 0x8fu,
|
||||||
0x92u, 0x96u, 0x9bu, 0x9cu, 0x9eu, 0x9fu, 0xa4u, 0xa7u, 0xa8u, 0xa9u, 0xaau, 0xabu,
|
0x92u, 0x96u, 0x9bu, 0x9cu, 0x9eu, 0x9fu, 0xa4u, 0xa7u, 0xa8u, 0xa9u, 0xaau, 0xabu,
|
||||||
@ -53,7 +53,7 @@ class C128Zeropage(options: CompilationOptions) : Zeropage(options) {
|
|||||||
|
|
||||||
// if(options.zeropage==ZeropageType.BASICSAFE) {
|
// if(options.zeropage==ZeropageType.BASICSAFE) {
|
||||||
// can also clobber the FP locations (unconditionally, because the C128 target doesn't support floating point calculations in prog8 at this time0
|
// can also clobber the FP locations (unconditionally, because the C128 target doesn't support floating point calculations in prog8 at this time0
|
||||||
free.addAll(listOf(0x14u, 0x28u, 0x29u, 0x2au, 0x2bu, 0x2cu,
|
free.addAll(arrayOf(0x14u, 0x28u, 0x29u, 0x2au, 0x2bu, 0x2cu,
|
||||||
0x50u, 0x51u, 0x52u, 0x53u, 0x54u, 0x59u, 0x5au, 0x5bu, 0x5cu, 0x5du, 0x5eu, 0x5fu, 0x60u, 0x61u, 0x62u,
|
0x50u, 0x51u, 0x52u, 0x53u, 0x54u, 0x59u, 0x5au, 0x5bu, 0x5cu, 0x5du, 0x5eu, 0x5fu, 0x60u, 0x61u, 0x62u,
|
||||||
0x63u, 0x64u, 0x65u, 0x66u, 0x67u, 0x68u,
|
0x63u, 0x64u, 0x65u, 0x66u, 0x67u, 0x68u,
|
||||||
0x6au, 0x6bu, 0x6cu, 0x6du, 0x6eu, 0x6fu, 0x71u))
|
0x6au, 0x6bu, 0x6cu, 0x6du, 0x6eu, 0x6fu, 0x71u))
|
||||||
|
@ -14,10 +14,13 @@ class C64MachineDefinition: IMachineDefinition {
|
|||||||
override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE
|
override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE
|
||||||
override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE
|
override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE
|
||||||
override val FLOAT_MEM_SIZE = Mflpt5.FLOAT_MEM_SIZE
|
override val FLOAT_MEM_SIZE = Mflpt5.FLOAT_MEM_SIZE
|
||||||
|
override val STARTUP_CODE_RESERVED_SIZE = 20u
|
||||||
override val PROGRAM_LOAD_ADDRESS = 0x0801u
|
override val PROGRAM_LOAD_ADDRESS = 0x0801u
|
||||||
|
override val PROGRAM_MEMTOP_ADDRESS = 0xcfe0u // $a000 if floats are used
|
||||||
|
// note that at $cfe0-$cfff are the 16 'virtual registers' R0-R15
|
||||||
|
|
||||||
override val BSSHIGHRAM_START = 0xc000u
|
override val BSSHIGHRAM_START = 0xc000u
|
||||||
override val BSSHIGHRAM_END = 0xcfffu
|
override val BSSHIGHRAM_END = 0xcfdfu
|
||||||
override val BSSGOLDENRAM_START = 0u
|
override val BSSGOLDENRAM_START = 0u
|
||||||
override val BSSGOLDENRAM_END = 0u
|
override val BSSGOLDENRAM_END = 0u
|
||||||
|
|
||||||
@ -26,11 +29,15 @@ class C64MachineDefinition: IMachineDefinition {
|
|||||||
|
|
||||||
override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm()
|
override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm()
|
||||||
|
|
||||||
override fun importLibs(compilerOptions: CompilationOptions, compilationTargetName: String): List<String> {
|
override fun convertFloatToBytes(num: Double): List<UByte> {
|
||||||
return if (compilerOptions.launcher == CbmPrgLauncherType.BASIC || compilerOptions.output == OutputType.PRG)
|
val m5 = Mflpt5.fromNumber(num)
|
||||||
listOf("syslib")
|
return listOf(m5.b0, m5.b1, m5.b2, m5.b3, m5.b4)
|
||||||
else
|
}
|
||||||
emptyList()
|
|
||||||
|
override fun convertBytesToFloat(bytes: List<UByte>): Double {
|
||||||
|
require(bytes.size==5) { "need 5 bytes" }
|
||||||
|
val m5 = Mflpt5(bytes[0], bytes[1], bytes[2], bytes[3], bytes[4])
|
||||||
|
return m5.toDouble()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
||||||
@ -48,7 +55,7 @@ class C64MachineDefinition: IMachineDefinition {
|
|||||||
val process: Process
|
val process: Process
|
||||||
try {
|
try {
|
||||||
process=processb.start()
|
process=processb.start()
|
||||||
} catch(x: IOException) {
|
} catch(_: IOException) {
|
||||||
continue // try the next emulator executable
|
continue // try the next emulator executable
|
||||||
}
|
}
|
||||||
process.waitFor()
|
process.waitFor()
|
||||||
|
@ -21,11 +21,11 @@ class C64Zeropage(options: CompilationOptions) : Zeropage(options) {
|
|||||||
|
|
||||||
if (options.zeropage == ZeropageType.FULL) {
|
if (options.zeropage == ZeropageType.FULL) {
|
||||||
free.addAll(0x02u..0xffu)
|
free.addAll(0x02u..0xffu)
|
||||||
free.removeAll(setOf(SCRATCH_B1, SCRATCH_REG, SCRATCH_W1, SCRATCH_W1+1u, SCRATCH_W2, SCRATCH_W2+1u))
|
free.removeAll(arrayOf(SCRATCH_B1, SCRATCH_REG, SCRATCH_W1, SCRATCH_W1+1u, SCRATCH_W2, SCRATCH_W2+1u))
|
||||||
free.removeAll(setOf(0xa0u, 0xa1u, 0xa2u, 0x91u, 0xc0u, 0xc5u, 0xcbu, 0xf5u, 0xf6u)) // these are updated by IRQ
|
free.removeAll(arrayOf(0xa0u, 0xa1u, 0xa2u, 0x91u, 0xc0u, 0xc5u, 0xcbu, 0xf5u, 0xf6u)) // these are updated by IRQ
|
||||||
} else {
|
} else {
|
||||||
if (options.zeropage == ZeropageType.KERNALSAFE || options.zeropage == ZeropageType.FLOATSAFE) {
|
if (options.zeropage == ZeropageType.KERNALSAFE || options.zeropage == ZeropageType.FLOATSAFE) {
|
||||||
free.addAll(listOf(
|
free.addAll(arrayOf(
|
||||||
0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
|
0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
|
||||||
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
|
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17, 0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
|
||||||
0x20, 0x21, 0x22, 0x23, 0x24, 0x25,
|
0x20, 0x21, 0x22, 0x23, 0x24, 0x25,
|
||||||
@ -43,7 +43,7 @@ class C64Zeropage(options: CompilationOptions) : Zeropage(options) {
|
|||||||
|
|
||||||
if (options.zeropage == ZeropageType.FLOATSAFE) {
|
if (options.zeropage == ZeropageType.FLOATSAFE) {
|
||||||
// remove the zeropage locations used for floating point operations from the free list
|
// remove the zeropage locations used for floating point operations from the free list
|
||||||
free.removeAll(listOf(
|
free.removeAll(arrayOf(
|
||||||
0x03, 0x04, 0x05, 0x06, 0x10, 0x11, 0x12,
|
0x03, 0x04, 0x05, 0x06, 0x10, 0x11, 0x12,
|
||||||
0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a,
|
0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a,
|
||||||
0x57, 0x58, 0x59, 0x5a, 0x5b, 0x5c, 0x5d, 0x5e, 0x5f, 0x60,
|
0x57, 0x58, 0x59, 0x5a, 0x5b, 0x5c, 0x5d, 0x5e, 0x5f, 0x60,
|
||||||
@ -56,7 +56,7 @@ class C64Zeropage(options: CompilationOptions) : Zeropage(options) {
|
|||||||
if(options.zeropage != ZeropageType.DONTUSE) {
|
if(options.zeropage != ZeropageType.DONTUSE) {
|
||||||
// add the free Zp addresses
|
// add the free Zp addresses
|
||||||
// these are valid for the C-64 but allow BASIC to keep running fully *as long as you don't use tape I/O*
|
// these are valid for the C-64 but allow BASIC to keep running fully *as long as you don't use tape I/O*
|
||||||
free.addAll(listOf(0x02, 0x03, 0x04, 0x05, 0x06, 0x0a, 0x0e,
|
free.addAll(arrayOf(0x02, 0x03, 0x04, 0x05, 0x06, 0x0a, 0x0e,
|
||||||
0x92, 0x96, 0x9b, 0x9c, 0x9e, 0x9f, 0xa6,
|
0x92, 0x96, 0x9b, 0x9c, 0x9e, 0x9f, 0xa6,
|
||||||
0xb0, 0xb1, 0xbe, 0xbf, 0xf9).map{it.toUInt()})
|
0xb0, 0xb1, 0xbe, 0xbf, 0xf9).map{it.toUInt()})
|
||||||
} else {
|
} else {
|
||||||
@ -81,16 +81,16 @@ class C64Zeropage(options: CompilationOptions) : Zeropage(options) {
|
|||||||
|
|
||||||
override fun allocateCx16VirtualRegisters() {
|
override fun allocateCx16VirtualRegisters() {
|
||||||
// Note: the 16 virtual registers R0-R15 are not regular allocated variables, they're *memory mapped* elsewhere to fixed addresses.
|
// Note: the 16 virtual registers R0-R15 are not regular allocated variables, they're *memory mapped* elsewhere to fixed addresses.
|
||||||
// However, to be able for the compiler to "see" them as zero page variables, we have to register them here as well.
|
// However, to be able for the compiler to "see" them as zeropage variables, we have to register them here as well.
|
||||||
// This is important because the compiler sometimes treats ZP variables more efficiently (for example if it's a pointer)
|
// This is important because the compiler sometimes treats ZP variables more efficiently (for example if it's a pointer)
|
||||||
// The base addres is $04. Unfortunately it cannot be the same as on the Commander X16 ($02).
|
// The base addres is $04. Unfortunately it cannot be the same as on the Commander X16 ($02).
|
||||||
for(reg in 0..15) {
|
for(reg in 0..15) {
|
||||||
allocatedVariables["cx16.r${reg}"] = VarAllocation((4+reg*2).toUInt(), DataType.UWORD, 2) // cx16.r0 .. cx16.r15
|
allocatedVariables["cx16.r${reg}"] = VarAllocation((4+reg*2).toUInt(), DataType.forDt(BaseDataType.UWORD), 2) // cx16.r0 .. cx16.r15
|
||||||
allocatedVariables["cx16.r${reg}s"] = VarAllocation((4+reg*2).toUInt(), DataType.WORD, 2) // cx16.r0s .. cx16.r15s
|
allocatedVariables["cx16.r${reg}s"] = VarAllocation((4+reg*2).toUInt(), DataType.forDt(BaseDataType.WORD), 2) // cx16.r0s .. cx16.r15s
|
||||||
allocatedVariables["cx16.r${reg}L"] = VarAllocation((4+reg*2).toUInt(), DataType.UBYTE, 1) // cx16.r0L .. cx16.r15L
|
allocatedVariables["cx16.r${reg}L"] = VarAllocation((4+reg*2).toUInt(), DataType.forDt(BaseDataType.UBYTE), 1) // cx16.r0L .. cx16.r15L
|
||||||
allocatedVariables["cx16.r${reg}H"] = VarAllocation((5+reg*2).toUInt(), DataType.UBYTE, 1) // cx16.r0H .. cx16.r15H
|
allocatedVariables["cx16.r${reg}H"] = VarAllocation((5+reg*2).toUInt(), DataType.forDt(BaseDataType.UBYTE), 1) // cx16.r0H .. cx16.r15H
|
||||||
allocatedVariables["cx16.r${reg}sL"] = VarAllocation((4+reg*2).toUInt(), DataType.BYTE, 1) // cx16.r0sL .. cx16.r15sL
|
allocatedVariables["cx16.r${reg}sL"] = VarAllocation((4+reg*2).toUInt(), DataType.forDt(BaseDataType.BYTE), 1) // cx16.r0sL .. cx16.r15sL
|
||||||
allocatedVariables["cx16.r${reg}sH"] = VarAllocation((5+reg*2).toUInt(), DataType.BYTE, 1) // cx16.r0sH .. cx16.r15sH
|
allocatedVariables["cx16.r${reg}sH"] = VarAllocation((5+reg*2).toUInt(), DataType.forDt(BaseDataType.BYTE), 1) // cx16.r0sH .. cx16.r15sH
|
||||||
free.remove((4+reg*2).toUInt())
|
free.remove((4+reg*2).toUInt())
|
||||||
free.remove((5+reg*2).toUInt())
|
free.remove((5+reg*2).toUInt())
|
||||||
}
|
}
|
||||||
|
@ -1,21 +1,33 @@
|
|||||||
package prog8.code.target.cbm
|
package prog8.code.target.cbm
|
||||||
|
|
||||||
import prog8.code.core.*
|
import prog8.code.core.BaseDataType
|
||||||
|
import prog8.code.core.DataType
|
||||||
|
import prog8.code.core.IMemSizer
|
||||||
|
|
||||||
|
|
||||||
internal object CbmMemorySizer: IMemSizer {
|
internal object CbmMemorySizer: IMemSizer {
|
||||||
override fun memorySize(dt: DataType): Int {
|
override fun memorySize(dt: DataType, numElements: Int?): Int {
|
||||||
return when(dt) {
|
if(dt.isArray) {
|
||||||
in ByteDatatypesWithBoolean -> 1
|
if(numElements==null) return 2 // treat it as a pointer size
|
||||||
in WordDatatypes, in PassByReferenceDatatypes -> 2
|
return when(dt.sub) {
|
||||||
DataType.FLOAT -> Mflpt5.FLOAT_MEM_SIZE
|
BaseDataType.BOOL, BaseDataType.UBYTE, BaseDataType.BYTE -> numElements
|
||||||
else -> throw IllegalArgumentException("invalid datatype")
|
BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.STR -> numElements * 2
|
||||||
|
BaseDataType.FLOAT-> numElements * Mflpt5.FLOAT_MEM_SIZE
|
||||||
|
BaseDataType.UNDEFINED -> throw IllegalArgumentException("undefined has no memory size")
|
||||||
|
else -> throw IllegalArgumentException("invalid sub type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (dt.isString) {
|
||||||
|
if(numElements!=null) return numElements // treat it as the size of the given string with the length
|
||||||
|
else return 2 // treat it as the size to store a string pointer
|
||||||
|
}
|
||||||
|
|
||||||
|
return when {
|
||||||
|
dt.isByteOrBool -> 1 * (numElements ?: 1)
|
||||||
|
dt.isFloat -> Mflpt5.FLOAT_MEM_SIZE * (numElements ?: 1)
|
||||||
|
dt.isLong -> throw IllegalArgumentException("long can not yet be put into memory")
|
||||||
|
dt.isUndefined -> throw IllegalArgumentException("undefined has no memory size")
|
||||||
|
else -> 2 * (numElements ?: 1)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun memorySize(arrayDt: DataType, numElements: Int) =
|
|
||||||
if(arrayDt==DataType.UWORD)
|
|
||||||
numElements // pointer to bytes.
|
|
||||||
else
|
|
||||||
memorySize(ArrayToElementTypes.getValue(arrayDt)) * numElements
|
|
||||||
}
|
}
|
@ -13,7 +13,9 @@ class CX16MachineDefinition: IMachineDefinition {
|
|||||||
override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE
|
override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE
|
||||||
override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE
|
override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE
|
||||||
override val FLOAT_MEM_SIZE = Mflpt5.FLOAT_MEM_SIZE
|
override val FLOAT_MEM_SIZE = Mflpt5.FLOAT_MEM_SIZE
|
||||||
|
override val STARTUP_CODE_RESERVED_SIZE = 20u
|
||||||
override val PROGRAM_LOAD_ADDRESS = 0x0801u
|
override val PROGRAM_LOAD_ADDRESS = 0x0801u
|
||||||
|
override val PROGRAM_MEMTOP_ADDRESS = 0x9f00u
|
||||||
|
|
||||||
override val BSSHIGHRAM_START = 0xa000u // hiram bank 1, 8Kb, assumed to be active
|
override val BSSHIGHRAM_START = 0xa000u // hiram bank 1, 8Kb, assumed to be active
|
||||||
override val BSSHIGHRAM_END = 0xbfffu // Rom starts at $c000
|
override val BSSHIGHRAM_END = 0xbfffu // Rom starts at $c000
|
||||||
@ -24,11 +26,16 @@ class CX16MachineDefinition: IMachineDefinition {
|
|||||||
override lateinit var golden: GoldenRam
|
override lateinit var golden: GoldenRam
|
||||||
|
|
||||||
override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm()
|
override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm()
|
||||||
override fun importLibs(compilerOptions: CompilationOptions, compilationTargetName: String): List<String> {
|
|
||||||
return if (compilerOptions.launcher == CbmPrgLauncherType.BASIC || compilerOptions.output == OutputType.PRG)
|
override fun convertFloatToBytes(num: Double): List<UByte> {
|
||||||
listOf("syslib")
|
val m5 = Mflpt5.fromNumber(num)
|
||||||
else
|
return listOf(m5.b0, m5.b1, m5.b2, m5.b3, m5.b4)
|
||||||
emptyList()
|
}
|
||||||
|
|
||||||
|
override fun convertBytesToFloat(bytes: List<UByte>): Double {
|
||||||
|
require(bytes.size==5) { "need 5 bytes" }
|
||||||
|
val m5 = Mflpt5(bytes[0], bytes[1], bytes[2], bytes[3], bytes[4])
|
||||||
|
return m5.toDouble()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
||||||
|
@ -54,15 +54,15 @@ class CX16Zeropage(options: CompilationOptions) : Zeropage(options) {
|
|||||||
|
|
||||||
override fun allocateCx16VirtualRegisters() {
|
override fun allocateCx16VirtualRegisters() {
|
||||||
// Note: the 16 virtual registers R0-R15 are not regular allocated variables, they're *memory mapped* elsewhere to fixed addresses.
|
// Note: the 16 virtual registers R0-R15 are not regular allocated variables, they're *memory mapped* elsewhere to fixed addresses.
|
||||||
// However, to be able for the compiler to "see" them as zero page variables, we have to register them here as well.
|
// However, to be able for the compiler to "see" them as zeropage variables, we have to register them here as well.
|
||||||
// This is important because the compiler sometimes treats ZP variables more efficiently (for example if it's a pointer)
|
// This is important because the compiler sometimes treats ZP variables more efficiently (for example if it's a pointer)
|
||||||
for(reg in 0..15) {
|
for(reg in 0..15) {
|
||||||
allocatedVariables["cx16.r${reg}"] = VarAllocation((2+reg*2).toUInt(), DataType.UWORD, 2) // cx16.r0 .. cx16.r15
|
allocatedVariables["cx16.r${reg}"] = VarAllocation((2+reg*2).toUInt(), DataType.forDt(BaseDataType.UWORD), 2) // cx16.r0 .. cx16.r15
|
||||||
allocatedVariables["cx16.r${reg}s"] = VarAllocation((2+reg*2).toUInt(), DataType.WORD, 2) // cx16.r0s .. cx16.r15s
|
allocatedVariables["cx16.r${reg}s"] = VarAllocation((2+reg*2).toUInt(), DataType.forDt(BaseDataType.WORD), 2) // cx16.r0s .. cx16.r15s
|
||||||
allocatedVariables["cx16.r${reg}L"] = VarAllocation((2+reg*2).toUInt(), DataType.UBYTE, 1) // cx16.r0L .. cx16.r15L
|
allocatedVariables["cx16.r${reg}L"] = VarAllocation((2+reg*2).toUInt(), DataType.forDt(BaseDataType.UBYTE), 1) // cx16.r0L .. cx16.r15L
|
||||||
allocatedVariables["cx16.r${reg}H"] = VarAllocation((3+reg*2).toUInt(), DataType.UBYTE, 1) // cx16.r0H .. cx16.r15H
|
allocatedVariables["cx16.r${reg}H"] = VarAllocation((3+reg*2).toUInt(), DataType.forDt(BaseDataType.UBYTE), 1) // cx16.r0H .. cx16.r15H
|
||||||
allocatedVariables["cx16.r${reg}sL"] = VarAllocation((2+reg*2).toUInt(), DataType.BYTE, 1) // cx16.r0sL .. cx16.r15sL
|
allocatedVariables["cx16.r${reg}sL"] = VarAllocation((2+reg*2).toUInt(), DataType.forDt(BaseDataType.BYTE), 1) // cx16.r0sL .. cx16.r15sL
|
||||||
allocatedVariables["cx16.r${reg}sH"] = VarAllocation((3+reg*2).toUInt(), DataType.BYTE, 1) // cx16.r0sH .. cx16.r15sH
|
allocatedVariables["cx16.r${reg}sH"] = VarAllocation((3+reg*2).toUInt(), DataType.forDt(BaseDataType.BYTE), 1) // cx16.r0sH .. cx16.r15sH
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1,4 +1,4 @@
|
|||||||
package prog8.code.target.cbm
|
package prog8.code.target.encodings
|
||||||
|
|
||||||
import com.github.michaelbull.result.Ok
|
import com.github.michaelbull.result.Ok
|
||||||
import com.github.michaelbull.result.Result
|
import com.github.michaelbull.result.Result
|
69
codeCore/src/prog8/code/target/encodings/Cp437Encoding.kt
Normal file
69
codeCore/src/prog8/code/target/encodings/Cp437Encoding.kt
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
package prog8.code.target.encodings
|
||||||
|
|
||||||
|
import com.github.michaelbull.result.Err
|
||||||
|
import com.github.michaelbull.result.Ok
|
||||||
|
import com.github.michaelbull.result.Result
|
||||||
|
import java.io.CharConversionException
|
||||||
|
import java.nio.charset.Charset
|
||||||
|
|
||||||
|
object Cp437Encoding {
|
||||||
|
val charset: Charset = Charset.forName("IBM437")
|
||||||
|
|
||||||
|
fun encode(str: String): Result<List<UByte>, CharConversionException> {
|
||||||
|
return try {
|
||||||
|
val mapped = str.map { chr ->
|
||||||
|
when (chr) {
|
||||||
|
'\u0000' -> 0u
|
||||||
|
'\u00a0' -> 255u
|
||||||
|
'☺' -> 1u
|
||||||
|
'☻' -> 2u
|
||||||
|
'♥' -> 3u
|
||||||
|
'♦' -> 4u
|
||||||
|
'♣' -> 5u
|
||||||
|
'♠' -> 6u
|
||||||
|
'•' -> 7u
|
||||||
|
'◘' -> 8u
|
||||||
|
'○' -> 9u
|
||||||
|
'◙' -> 10u
|
||||||
|
'♂' -> 11u
|
||||||
|
'♀' -> 12u
|
||||||
|
'♪' -> 13u
|
||||||
|
'♫' -> 14u
|
||||||
|
'☼' -> 15u
|
||||||
|
'►' -> 16u
|
||||||
|
'◄' -> 17u
|
||||||
|
'↕' -> 18u
|
||||||
|
'‼' -> 19u
|
||||||
|
'¶' -> 20u
|
||||||
|
'§' -> 21u
|
||||||
|
'▬' -> 22u
|
||||||
|
'↨' -> 23u
|
||||||
|
'↑' -> 24u
|
||||||
|
'↓' -> 25u
|
||||||
|
'→' -> 26u
|
||||||
|
'←' -> 27u
|
||||||
|
'∟' -> 28u
|
||||||
|
'↔' -> 29u
|
||||||
|
'▲' -> 30u
|
||||||
|
'▼' -> 31u
|
||||||
|
in '\u8000'..'\u80ff' -> {
|
||||||
|
// special case: take the lower 8 bit hex value directly
|
||||||
|
(chr.code - 0x8000).toUByte()
|
||||||
|
}
|
||||||
|
else -> charset.encode(chr.toString())[0].toUByte()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(mapped)
|
||||||
|
} catch (ce: CharConversionException) {
|
||||||
|
Err(ce)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun decode(bytes: Iterable<UByte>): Result<String, CharConversionException> {
|
||||||
|
return try {
|
||||||
|
Ok(String(bytes.map { it.toByte() }.toByteArray(), charset))
|
||||||
|
} catch (ce: CharConversionException) {
|
||||||
|
Err(ce)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
package prog8.code.target.cbm
|
package prog8.code.target.encodings
|
||||||
|
|
||||||
import com.github.michaelbull.result.Err
|
import com.github.michaelbull.result.Err
|
||||||
import com.github.michaelbull.result.Ok
|
import com.github.michaelbull.result.Ok
|
||||||
@ -6,8 +6,8 @@ import com.github.michaelbull.result.Result
|
|||||||
import java.io.CharConversionException
|
import java.io.CharConversionException
|
||||||
import java.nio.charset.Charset
|
import java.nio.charset.Charset
|
||||||
|
|
||||||
object IsoEncoding {
|
open class IsoEncodingBase(charsetName: String) {
|
||||||
val charset: Charset = Charset.forName("ISO-8859-15")
|
val charset: Charset = Charset.forName(charsetName)
|
||||||
|
|
||||||
fun encode(str: String): Result<List<UByte>, CharConversionException> {
|
fun encode(str: String): Result<List<UByte>, CharConversionException> {
|
||||||
return try {
|
return try {
|
||||||
@ -35,3 +35,8 @@ object IsoEncoding {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
object IsoEncoding: IsoEncodingBase("ISO-8859-15")
|
||||||
|
object IsoCyrillicEncoding: IsoEncodingBase("ISO-8859-5")
|
||||||
|
object IsoEasternEncoding: IsoEncodingBase("ISO-8859-16")
|
122
codeCore/src/prog8/code/target/encodings/KatakanaEncoding.kt
Normal file
122
codeCore/src/prog8/code/target/encodings/KatakanaEncoding.kt
Normal file
@ -0,0 +1,122 @@
|
|||||||
|
package prog8.code.target.encodings
|
||||||
|
|
||||||
|
import com.github.michaelbull.result.Err
|
||||||
|
import com.github.michaelbull.result.Ok
|
||||||
|
import com.github.michaelbull.result.Result
|
||||||
|
import java.io.CharConversionException
|
||||||
|
import java.nio.charset.Charset
|
||||||
|
|
||||||
|
|
||||||
|
object JapaneseCharacterConverter {
|
||||||
|
// adapted from https://github.com/raminduw/Japanese-Character-Converter
|
||||||
|
|
||||||
|
private val ZENKAKU_KATAKANA = charArrayOf(
|
||||||
|
'ァ', 'ア', 'ィ', 'イ', 'ゥ',
|
||||||
|
'ウ', 'ェ', 'エ', 'ォ', 'オ', 'カ', 'ガ', 'キ', 'ギ', 'ク', 'グ', 'ケ', 'ゲ',
|
||||||
|
'コ', 'ゴ', 'サ', 'ザ', 'シ', 'ジ', 'ス', 'ズ', 'セ', 'ゼ', 'ソ', 'ゾ', 'タ',
|
||||||
|
'ダ', 'チ', 'ヂ', 'ッ', 'ツ', 'ヅ', 'テ', 'デ', 'ト', 'ド', 'ナ', 'ニ', 'ヌ',
|
||||||
|
'ネ', 'ノ', 'ハ', 'バ', 'パ', 'ヒ', 'ビ', 'ピ', 'フ', 'ブ', 'プ', 'ヘ', 'ベ',
|
||||||
|
'ペ', 'ホ', 'ボ', 'ポ', 'マ', 'ミ', 'ム', 'メ', 'モ', 'ャ', 'ヤ', 'ュ', 'ユ',
|
||||||
|
'ョ', 'ヨ', 'ラ', 'リ', 'ル', 'レ', 'ロ', 'ヮ', 'ワ', 'ヰ', 'ヱ', 'ヲ', 'ン',
|
||||||
|
'ヴ', 'ヵ', 'ヶ'
|
||||||
|
)
|
||||||
|
|
||||||
|
private val HANKAKU_HIRAGANA = charArrayOf(
|
||||||
|
'ぁ', 'あ', 'ぃ', 'い', 'ぅ', 'う', 'ぇ', 'え',
|
||||||
|
'ぉ', 'お', 'か', 'が', 'き', 'ぎ', 'く', 'ぐ',
|
||||||
|
'け', 'げ', 'こ', 'ご', 'さ', 'ざ', 'し', 'じ',
|
||||||
|
'す', 'ず', 'せ', 'ぜ', 'そ', 'ぞ', 'た', 'だ',
|
||||||
|
'ち', 'ぢ', 'っ', 'つ', 'づ', 'て', 'で', 'と',
|
||||||
|
'ど', 'な', 'に', 'ぬ', 'ね', 'の', 'は', 'ば',
|
||||||
|
'ぱ', 'ひ', 'び', 'ぴ', 'ふ', 'ぶ', 'ぷ', 'へ',
|
||||||
|
'べ', 'ぺ', 'ほ', 'ぼ', 'ぽ', 'ま', 'み', 'む',
|
||||||
|
'め', 'も', 'ゃ', 'や', 'ゅ', 'ゆ', 'ょ', 'よ',
|
||||||
|
'ら', 'り', 'る', 'れ', 'ろ', 'ゎ', 'わ', 'ゐ',
|
||||||
|
'ゑ', 'を', 'ん', 'ゔ', 'ゕ', 'ゖ'
|
||||||
|
)
|
||||||
|
|
||||||
|
private val HANKAKU_KATAKANA = arrayOf(
|
||||||
|
"ァ", "ア", "ィ", "イ", "ゥ",
|
||||||
|
"ウ", "ェ", "エ", "ォ", "オ", "カ", "ガ", "キ", "ギ", "ク", "グ", "ケ",
|
||||||
|
"ゲ", "コ", "ゴ", "サ", "ザ", "シ", "ジ", "ス", "ズ", "セ", "ゼ", "ソ",
|
||||||
|
"ゾ", "タ", "ダ", "チ", "ヂ", "ッ", "ツ", "ヅ", "テ", "デ", "ト", "ド",
|
||||||
|
"ナ", "ニ", "ヌ", "ネ", "ノ", "ハ", "バ", "パ", "ヒ", "ビ", "ピ", "フ",
|
||||||
|
"ブ", "プ", "ヘ", "ベ", "ペ", "ホ", "ボ", "ポ", "マ", "ミ", "ム", "メ",
|
||||||
|
"モ", "ャ", "ヤ", "ュ", "ユ", "ョ", "ヨ", "ラ", "リ", "ル", "レ", "ロ", "ワ",
|
||||||
|
"ワ", "イ", "エ", "ヲ", "ン", "ヴ", "カ", "ケ"
|
||||||
|
)
|
||||||
|
|
||||||
|
private val ZENKAKU_KATAKANA_FIRST_CHAR_CODE = ZENKAKU_KATAKANA.first().code
|
||||||
|
private val HANKAKU_HIRAGANA_FIRST_CHAR_CODE = HANKAKU_HIRAGANA.first().code
|
||||||
|
|
||||||
|
private fun zenkakuKatakanaToHankakuKatakana(c: Char): String = if (c in ZENKAKU_KATAKANA) HANKAKU_KATAKANA[c.code - ZENKAKU_KATAKANA_FIRST_CHAR_CODE] else c.toString()
|
||||||
|
private fun hankakuKatakanaToZenkakuKatakana(c: Char): Char = if (c in HANKAKU_HIRAGANA) ZENKAKU_KATAKANA[c.code - HANKAKU_HIRAGANA_FIRST_CHAR_CODE] else c
|
||||||
|
|
||||||
|
fun zenkakuKatakanaToHankakuKatakana(s: String): String = buildString {
|
||||||
|
for (element in s) {
|
||||||
|
val converted = hankakuKatakanaToZenkakuKatakana(element)
|
||||||
|
val convertedChar = zenkakuKatakanaToHankakuKatakana(converted)
|
||||||
|
append(convertedChar)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
object KatakanaEncoding {
|
||||||
|
val charset: Charset = Charset.forName("JIS_X0201")
|
||||||
|
|
||||||
|
fun encode(str: String): Result<List<UByte>, CharConversionException> {
|
||||||
|
return try {
|
||||||
|
val mapped = str.map { chr ->
|
||||||
|
when (chr) {
|
||||||
|
|
||||||
|
'\u0000' -> 0u
|
||||||
|
'\u00a0' -> 0xa0u // $a0 isn't technically a part of JIS X 0201 spec, and so we need to handle this ourselves
|
||||||
|
|
||||||
|
'♥' -> 0xe3u
|
||||||
|
'♦' -> 0xe4u
|
||||||
|
'♣' -> 0xe5u
|
||||||
|
'♠' -> 0xe6u
|
||||||
|
|
||||||
|
'大' -> 0xeau
|
||||||
|
'中' -> 0xebu
|
||||||
|
'小' -> 0xecu
|
||||||
|
'百' -> 0xedu
|
||||||
|
'千' -> 0xeeu
|
||||||
|
'万' -> 0xefu
|
||||||
|
'♪' -> 0xf0u
|
||||||
|
'土' -> 0xf1u
|
||||||
|
'金' -> 0xf2u
|
||||||
|
'木' -> 0xf3u
|
||||||
|
'水' -> 0xf4u
|
||||||
|
'火' -> 0xf5u
|
||||||
|
'月' -> 0xf6u
|
||||||
|
'日' -> 0xf7u
|
||||||
|
'時' -> 0xf8u
|
||||||
|
'分' -> 0xf9u
|
||||||
|
'秒' -> 0xfau
|
||||||
|
'年' -> 0xfbu
|
||||||
|
'円' -> 0xfcu
|
||||||
|
'人' -> 0xfdu
|
||||||
|
'生' -> 0xfeu
|
||||||
|
'〒' -> 0xffu
|
||||||
|
in '\u8000'..'\u80ff' -> {
|
||||||
|
// special case: take the lower 8 bit hex value directly
|
||||||
|
(chr.code - 0x8000).toUByte()
|
||||||
|
}
|
||||||
|
else -> charset.encode(chr.toString())[0].toUByte()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(mapped)
|
||||||
|
} catch (ce: CharConversionException) {
|
||||||
|
Err(ce)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fun decode(bytes: Iterable<UByte>): Result<String, CharConversionException> {
|
||||||
|
return try {
|
||||||
|
Ok(String(bytes.map { it.toByte() }.toByteArray(), charset))
|
||||||
|
} catch (ce: CharConversionException) {
|
||||||
|
Err(ce)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
package prog8.code.target.cbm
|
package prog8.code.target.encodings
|
||||||
|
|
||||||
import com.github.michaelbull.result.Err
|
import com.github.michaelbull.result.Err
|
||||||
import com.github.michaelbull.result.Ok
|
import com.github.michaelbull.result.Ok
|
||||||
@ -7,7 +7,7 @@ import java.io.CharConversionException
|
|||||||
|
|
||||||
object PetsciiEncoding {
|
object PetsciiEncoding {
|
||||||
|
|
||||||
// decoding: from Petscii/Screencodes (0-255) to unicode
|
// decoding: from Petscii/Screencodes (0-255) to Unicode
|
||||||
// character tables used from https://github.com/irmen/cbmcodecs2
|
// character tables used from https://github.com/irmen/cbmcodecs2
|
||||||
|
|
||||||
private val decodingPetsciiLowercase = charArrayOf(
|
private val decodingPetsciiLowercase = charArrayOf(
|
||||||
@ -1089,7 +1089,7 @@ object PetsciiEncoding {
|
|||||||
Ok(text.map {
|
Ok(text.map {
|
||||||
try {
|
try {
|
||||||
encodeChar(it, lowercase)
|
encodeChar(it, lowercase)
|
||||||
} catch (x: CharConversionException) {
|
} catch (_: CharConversionException) {
|
||||||
encodeChar(it, !lowercase)
|
encodeChar(it, !lowercase)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@ -1135,7 +1135,7 @@ object PetsciiEncoding {
|
|||||||
Ok(text.map {
|
Ok(text.map {
|
||||||
try {
|
try {
|
||||||
encodeChar(it, lowercase)
|
encodeChar(it, lowercase)
|
||||||
} catch (x: CharConversionException) {
|
} catch (_: CharConversionException) {
|
||||||
encodeChar(it, !lowercase)
|
encodeChar(it, !lowercase)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
@ -1157,16 +1157,16 @@ object PetsciiEncoding {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fun petscii2scr(petscii_code: UByte, inverseVideo: Boolean): Result<UByte, CharConversionException> {
|
fun petscii2scr(petsciicode: UByte, inverseVideo: Boolean): Result<UByte, CharConversionException> {
|
||||||
val code: UInt = when {
|
val code: UInt = when {
|
||||||
petscii_code <= 0x1fu -> petscii_code + 128u
|
petsciicode <= 0x1fu -> petsciicode + 128u
|
||||||
petscii_code <= 0x3fu -> petscii_code.toUInt()
|
petsciicode <= 0x3fu -> petsciicode.toUInt()
|
||||||
petscii_code <= 0x5fu -> petscii_code - 64u
|
petsciicode <= 0x5fu -> petsciicode - 64u
|
||||||
petscii_code <= 0x7fu -> petscii_code - 32u
|
petsciicode <= 0x7fu -> petsciicode - 32u
|
||||||
petscii_code <= 0x9fu -> petscii_code + 64u
|
petsciicode <= 0x9fu -> petsciicode + 64u
|
||||||
petscii_code <= 0xbfu -> petscii_code - 64u
|
petsciicode <= 0xbfu -> petsciicode - 64u
|
||||||
petscii_code <= 0xfeu -> petscii_code - 128u
|
petsciicode <= 0xfeu -> petsciicode - 128u
|
||||||
petscii_code == 255.toUByte() -> 95u
|
petsciicode == 255.toUByte() -> 95u
|
||||||
else -> return Err(CharConversionException("petscii code out of range"))
|
else -> return Err(CharConversionException("petscii code out of range"))
|
||||||
}
|
}
|
||||||
if(inverseVideo) {
|
if(inverseVideo) {
|
@ -0,0 +1,50 @@
|
|||||||
|
package prog8.code.target.neo6502
|
||||||
|
|
||||||
|
import prog8.code.core.*
|
||||||
|
import java.nio.file.Path
|
||||||
|
|
||||||
|
|
||||||
|
class Neo6502MachineDefinition: IMachineDefinition {
|
||||||
|
|
||||||
|
override val cpu = CpuType.CPU65c02
|
||||||
|
|
||||||
|
override val FLOAT_MAX_POSITIVE = 9.999999999e97
|
||||||
|
override val FLOAT_MAX_NEGATIVE = -9.999999999e97
|
||||||
|
override val FLOAT_MEM_SIZE = 6
|
||||||
|
override val STARTUP_CODE_RESERVED_SIZE = 20u
|
||||||
|
override val PROGRAM_LOAD_ADDRESS = 0x0800u
|
||||||
|
override val PROGRAM_MEMTOP_ADDRESS = 0xfc00u // kernal starts here
|
||||||
|
|
||||||
|
override val BSSHIGHRAM_START = 0u // TODO
|
||||||
|
override val BSSHIGHRAM_END = 0u // TODO
|
||||||
|
override val BSSGOLDENRAM_START = 0u // TODO
|
||||||
|
override val BSSGOLDENRAM_END = 0u // TODO
|
||||||
|
|
||||||
|
override lateinit var zeropage: Zeropage
|
||||||
|
override lateinit var golden: GoldenRam
|
||||||
|
|
||||||
|
override fun getFloatAsmBytes(num: Number) = TODO("atari float asm bytes from number")
|
||||||
|
override fun convertFloatToBytes(num: Double): List<UByte> = TODO("atari float to bytes")
|
||||||
|
override fun convertBytesToFloat(bytes: List<UByte>): Double = TODO("atari bytes to float")
|
||||||
|
|
||||||
|
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
||||||
|
if(selectedEmulator!=1) {
|
||||||
|
System.err.println("The neo target only supports the main emulator (neo).")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
val cmdline = listOf("neo", "${programNameWithPath}.bin@800", "cold")
|
||||||
|
|
||||||
|
println("\nStarting Neo6502 emulator...")
|
||||||
|
val processb = ProcessBuilder(cmdline).inheritIO()
|
||||||
|
val process: Process = processb.start()
|
||||||
|
process.waitFor()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun isIOAddress(address: UInt): Boolean = address in 0xff00u..0xff0fu
|
||||||
|
|
||||||
|
override fun initializeMemoryAreas(compilerOptions: CompilationOptions) {
|
||||||
|
zeropage = Neo6502Zeropage(compilerOptions)
|
||||||
|
golden = GoldenRam(compilerOptions, UIntRange.EMPTY)
|
||||||
|
}
|
||||||
|
}
|
48
codeCore/src/prog8/code/target/neo6502/Neo6502Zeropage.kt
Normal file
48
codeCore/src/prog8/code/target/neo6502/Neo6502Zeropage.kt
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
package prog8.code.target.neo6502
|
||||||
|
|
||||||
|
import prog8.code.core.*
|
||||||
|
|
||||||
|
class Neo6502Zeropage(options: CompilationOptions) : Zeropage(options) {
|
||||||
|
|
||||||
|
override val SCRATCH_B1 = 0xfau // temp storage for a single byte
|
||||||
|
override val SCRATCH_REG = 0xfbu // temp storage for a register, must be B1+1
|
||||||
|
override val SCRATCH_W1 = 0xfcu // temp storage 1 for a word $fc+$fd
|
||||||
|
override val SCRATCH_W2 = 0xfeu // temp storage 2 for a word $fe+$ff
|
||||||
|
|
||||||
|
init {
|
||||||
|
if (options.floats) {
|
||||||
|
throw InternalCompilerException("Neo6502 target doesn't support floating point routines")
|
||||||
|
}
|
||||||
|
|
||||||
|
when (options.zeropage) {
|
||||||
|
ZeropageType.DONTUSE -> {
|
||||||
|
free.clear() // don't use zeropage at all
|
||||||
|
}
|
||||||
|
else -> {
|
||||||
|
free.addAll(0x22u..0xffu)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
val distinctFree = free.distinct()
|
||||||
|
free.clear()
|
||||||
|
free.addAll(distinctFree)
|
||||||
|
|
||||||
|
removeReservedFromFreePool()
|
||||||
|
allocateCx16VirtualRegisters()
|
||||||
|
retainAllowed()
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun allocateCx16VirtualRegisters() {
|
||||||
|
// Note: the 16 virtual registers R0-R15 are not regular allocated variables, they're *memory mapped* elsewhere to fixed addresses.
|
||||||
|
// However, to be able for the compiler to "see" them as zeropage variables, we have to register them here as well.
|
||||||
|
// This is important because the compiler sometimes treats ZP variables more efficiently (for example if it's a pointer)
|
||||||
|
for(reg in 0..15) {
|
||||||
|
allocatedVariables["cx16.r${reg}"] = VarAllocation((2+reg*2).toUInt(), DataType.forDt(BaseDataType.UWORD), 2) // cx16.r0 .. cx16.r15
|
||||||
|
allocatedVariables["cx16.r${reg}s"] = VarAllocation((2+reg*2).toUInt(), DataType.forDt(BaseDataType.WORD), 2) // cx16.r0s .. cx16.r15s
|
||||||
|
allocatedVariables["cx16.r${reg}L"] = VarAllocation((2+reg*2).toUInt(), DataType.forDt(BaseDataType.UBYTE), 1) // cx16.r0L .. cx16.r15L
|
||||||
|
allocatedVariables["cx16.r${reg}H"] = VarAllocation((3+reg*2).toUInt(), DataType.forDt(BaseDataType.UBYTE), 1) // cx16.r0H .. cx16.r15H
|
||||||
|
allocatedVariables["cx16.r${reg}sL"] = VarAllocation((2+reg*2).toUInt(), DataType.forDt(BaseDataType.BYTE), 1) // cx16.r0sL .. cx16.r15sL
|
||||||
|
allocatedVariables["cx16.r${reg}sH"] = VarAllocation((3+reg*2).toUInt(), DataType.forDt(BaseDataType.BYTE), 1) // cx16.r0sH .. cx16.r15sH
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -13,7 +13,9 @@ class PETMachineDefinition: IMachineDefinition {
|
|||||||
override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE
|
override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE
|
||||||
override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE
|
override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE
|
||||||
override val FLOAT_MEM_SIZE = Mflpt5.FLOAT_MEM_SIZE
|
override val FLOAT_MEM_SIZE = Mflpt5.FLOAT_MEM_SIZE
|
||||||
|
override val STARTUP_CODE_RESERVED_SIZE = 20u
|
||||||
override val PROGRAM_LOAD_ADDRESS = 0x0401u
|
override val PROGRAM_LOAD_ADDRESS = 0x0401u
|
||||||
|
override val PROGRAM_MEMTOP_ADDRESS = 0x8000u
|
||||||
|
|
||||||
override val BSSHIGHRAM_START = 0u
|
override val BSSHIGHRAM_START = 0u
|
||||||
override val BSSHIGHRAM_END = 0u
|
override val BSSHIGHRAM_END = 0u
|
||||||
@ -25,11 +27,15 @@ class PETMachineDefinition: IMachineDefinition {
|
|||||||
|
|
||||||
override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm()
|
override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm()
|
||||||
|
|
||||||
override fun importLibs(compilerOptions: CompilationOptions, compilationTargetName: String): List<String> {
|
override fun convertFloatToBytes(num: Double): List<UByte> {
|
||||||
return if (compilerOptions.launcher == CbmPrgLauncherType.BASIC || compilerOptions.output == OutputType.PRG)
|
val m5 = Mflpt5.fromNumber(num)
|
||||||
listOf("syslib")
|
return listOf(m5.b0, m5.b1, m5.b2, m5.b3, m5.b4)
|
||||||
else
|
}
|
||||||
emptyList()
|
|
||||||
|
override fun convertBytesToFloat(bytes: List<UByte>): Double {
|
||||||
|
require(bytes.size==5) { "need 5 bytes" }
|
||||||
|
val m5 = Mflpt5(bytes[0], bytes[1], bytes[2], bytes[3], bytes[4])
|
||||||
|
return m5.toDouble()
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
||||||
|
@ -10,10 +10,12 @@ class VirtualMachineDefinition: IMachineDefinition {
|
|||||||
|
|
||||||
override val cpu = CpuType.VIRTUAL
|
override val cpu = CpuType.VIRTUAL
|
||||||
|
|
||||||
override val FLOAT_MAX_POSITIVE = Float.MAX_VALUE.toDouble()
|
override val FLOAT_MAX_POSITIVE = Double.MAX_VALUE.toDouble()
|
||||||
override val FLOAT_MAX_NEGATIVE = -Float.MAX_VALUE.toDouble()
|
override val FLOAT_MAX_NEGATIVE = -Double.MAX_VALUE.toDouble()
|
||||||
override val FLOAT_MEM_SIZE = 8 // 64-bits double
|
override val FLOAT_MEM_SIZE = 8 // 64-bits double
|
||||||
|
override val STARTUP_CODE_RESERVED_SIZE = 0u // not actually used
|
||||||
override val PROGRAM_LOAD_ADDRESS = 0u // not actually used
|
override val PROGRAM_LOAD_ADDRESS = 0u // not actually used
|
||||||
|
override val PROGRAM_MEMTOP_ADDRESS = 0xffffu // not actually used
|
||||||
|
|
||||||
override val BSSHIGHRAM_START = 0u // not actually used
|
override val BSSHIGHRAM_START = 0u // not actually used
|
||||||
override val BSSHIGHRAM_END = 0u // not actually used
|
override val BSSHIGHRAM_END = 0u // not actually used
|
||||||
@ -30,8 +32,24 @@ class VirtualMachineDefinition: IMachineDefinition {
|
|||||||
return parts.joinToString(", ")
|
return parts.joinToString(", ")
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun importLibs(compilerOptions: CompilationOptions, compilationTargetName: String): List<String> {
|
override fun convertFloatToBytes(num: Double): List<UByte> {
|
||||||
return listOf("syslib")
|
val bits = num.toBits().toULong()
|
||||||
|
val hexStr = bits.toString(16).padStart(16, '0')
|
||||||
|
val parts = hexStr.chunked(2).map { it.toInt(16).toUByte() }
|
||||||
|
return parts
|
||||||
|
}
|
||||||
|
|
||||||
|
override fun convertBytesToFloat(bytes: List<UByte>): Double {
|
||||||
|
require(bytes.size==8) { "need 8 bytes" }
|
||||||
|
val b0 = bytes[0].toLong() shl (8*7)
|
||||||
|
val b1 = bytes[1].toLong() shl (8*6)
|
||||||
|
val b2 = bytes[2].toLong() shl (8*5)
|
||||||
|
val b3 = bytes[3].toLong() shl (8*4)
|
||||||
|
val b4 = bytes[4].toLong() shl (8*3)
|
||||||
|
val b5 = bytes[5].toLong() shl (8*2)
|
||||||
|
val b6 = bytes[6].toLong() shl (8*1)
|
||||||
|
val b7 = bytes[7].toLong() shl (8*0)
|
||||||
|
return Double.fromBits(b0 or b1 or b2 or b3 or b4 or b5 or b6 or b7)
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path) {
|
||||||
|
@ -1,63 +0,0 @@
|
|||||||
plugins {
|
|
||||||
id 'java'
|
|
||||||
id 'application'
|
|
||||||
id "org.jetbrains.kotlin.jvm"
|
|
||||||
}
|
|
||||||
|
|
||||||
java {
|
|
||||||
toolchain {
|
|
||||||
languageVersion = JavaLanguageVersion.of(javaVersion)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
compileKotlin {
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = javaVersion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
compileTestKotlin {
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = javaVersion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
dependencies {
|
|
||||||
implementation project(':codeCore')
|
|
||||||
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk8"
|
|
||||||
// implementation "org.jetbrains.kotlin:kotlin-reflect"
|
|
||||||
implementation "com.michael-bull.kotlin-result:kotlin-result-jvm:1.1.20"
|
|
||||||
|
|
||||||
testImplementation 'io.kotest:kotest-runner-junit5-jvm:5.8.0'
|
|
||||||
testImplementation 'org.junit.jupiter:junit-jupiter:5.9.2'
|
|
||||||
testRuntimeOnly 'org.junit.platform:junit-platform-launcher'
|
|
||||||
}
|
|
||||||
|
|
||||||
sourceSets {
|
|
||||||
main {
|
|
||||||
java {
|
|
||||||
srcDir "${project.projectDir}/src"
|
|
||||||
}
|
|
||||||
resources {
|
|
||||||
srcDir "${project.projectDir}/res"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
test {
|
|
||||||
java {
|
|
||||||
srcDir "${project.projectDir}/test"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
test {
|
|
||||||
// Enable JUnit 5 (Gradle 4.6+).
|
|
||||||
useJUnitPlatform()
|
|
||||||
|
|
||||||
// Always run tests, even when nothing changed.
|
|
||||||
dependsOn 'cleanTest'
|
|
||||||
|
|
||||||
// Show test results.
|
|
||||||
testLogging {
|
|
||||||
events "skipped", "failed"
|
|
||||||
}
|
|
||||||
}
|
|
46
codeGenCpu6502/build.gradle.kts
Normal file
46
codeGenCpu6502/build.gradle.kts
Normal file
@ -0,0 +1,46 @@
|
|||||||
|
import org.jetbrains.kotlin.gradle.dsl.JvmTarget
|
||||||
|
|
||||||
|
plugins {
|
||||||
|
kotlin("jvm")
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
implementation(project(":codeCore"))
|
||||||
|
// implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk8")
|
||||||
|
// implementation "org.jetbrains.kotlin:kotlin-reflect"
|
||||||
|
implementation("com.michael-bull.kotlin-result:kotlin-result-jvm:2.0.0")
|
||||||
|
|
||||||
|
testImplementation("io.kotest:kotest-runner-junit5-jvm:5.9.1")
|
||||||
|
testImplementation("io.kotest:kotest-framework-datatest:5.9.1")
|
||||||
|
testImplementation("org.junit.jupiter:junit-jupiter:5.9.1")
|
||||||
|
testRuntimeOnly("org.junit.platform:junit-platform-launcher")
|
||||||
|
}
|
||||||
|
|
||||||
|
sourceSets {
|
||||||
|
main {
|
||||||
|
java {
|
||||||
|
setSrcDirs(listOf("$projectDir/src"))
|
||||||
|
}
|
||||||
|
resources {
|
||||||
|
setSrcDirs(listOf("$projectDir/res"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
test {
|
||||||
|
java {
|
||||||
|
setSrcDirs(listOf("$projectDir/test"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
tasks.test {
|
||||||
|
// Enable JUnit 5 (Gradle 4.6+).
|
||||||
|
useJUnitPlatform()
|
||||||
|
|
||||||
|
// Always run tests, even when nothing changed.
|
||||||
|
dependsOn("cleanTest")
|
||||||
|
|
||||||
|
// Show test results.
|
||||||
|
testLogging {
|
||||||
|
events("skipped", "failed")
|
||||||
|
}
|
||||||
|
}
|
@ -14,5 +14,6 @@
|
|||||||
<orderEntry type="library" name="michael.bull.kotlin.result.jvm" level="project" />
|
<orderEntry type="library" name="michael.bull.kotlin.result.jvm" level="project" />
|
||||||
<orderEntry type="library" name="io.kotest.assertions.core.jvm" level="project" />
|
<orderEntry type="library" name="io.kotest.assertions.core.jvm" level="project" />
|
||||||
<orderEntry type="library" name="io.kotest.runner.junit5.jvm" level="project" />
|
<orderEntry type="library" name="io.kotest.runner.junit5.jvm" level="project" />
|
||||||
|
<orderEntry type="library" name="io.kotest.framework.datatest" level="project" />
|
||||||
</component>
|
</component>
|
||||||
</module>
|
</module>
|
File diff suppressed because it is too large
Load Diff
@ -3,6 +3,7 @@ package prog8.codegen.cpu6502
|
|||||||
import prog8.code.StConstant
|
import prog8.code.StConstant
|
||||||
import prog8.code.StMemVar
|
import prog8.code.StMemVar
|
||||||
import prog8.code.SymbolTable
|
import prog8.code.SymbolTable
|
||||||
|
import prog8.code.ast.PtLabel
|
||||||
import prog8.code.core.IMachineDefinition
|
import prog8.code.core.IMachineDefinition
|
||||||
|
|
||||||
|
|
||||||
@ -50,6 +51,13 @@ internal fun optimizeAssembly(lines: MutableList<String>, machine: IMachineDefin
|
|||||||
numberOfOptimizations++
|
numberOfOptimizations++
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mods = optimizeTSBtoRegularOr(linesByFour)
|
||||||
|
if(mods.isNotEmpty()) {
|
||||||
|
apply(mods, lines)
|
||||||
|
linesByFour = getLinesBy(lines, 4)
|
||||||
|
numberOfOptimizations++
|
||||||
|
}
|
||||||
|
|
||||||
var linesByFourteen = getLinesBy(lines, 14)
|
var linesByFourteen = getLinesBy(lines, 14)
|
||||||
mods = optimizeSameAssignments(linesByFourteen, machine, symbolTable)
|
mods = optimizeSameAssignments(linesByFourteen, machine, symbolTable)
|
||||||
if(mods.isNotEmpty()) {
|
if(mods.isNotEmpty()) {
|
||||||
@ -65,8 +73,6 @@ internal fun optimizeAssembly(lines: MutableList<String>, machine: IMachineDefin
|
|||||||
numberOfOptimizations++
|
numberOfOptimizations++
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO more assembly peephole optimizations
|
|
||||||
|
|
||||||
return numberOfOptimizations
|
return numberOfOptimizations
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -356,15 +362,16 @@ or *_afterif labels.
|
|||||||
This gets generated after certain if conditions, and only the branch instruction is needed in these cases.
|
This gets generated after certain if conditions, and only the branch instruction is needed in these cases.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
val autoLabelPrefix = PtLabel.GENERATED_LABEL_PREFIX
|
||||||
if(first=="beq +" && second=="lda #1" && third=="+") {
|
if(first=="beq +" && second=="lda #1" && third=="+") {
|
||||||
if((fourth.startsWith("beq label_") || fourth.startsWith("bne label_")) &&
|
if((fourth.startsWith("beq $autoLabelPrefix") || fourth.startsWith("bne $autoLabelPrefix")) &&
|
||||||
(fourth.endsWith("_shortcut") || fourth.endsWith("_afterif") || fourth.endsWith("_shortcut:") || fourth.endsWith("_afterif:"))) {
|
(fourth.endsWith("_shortcut") || fourth.endsWith("_afterif") || fourth.endsWith("_shortcut:") || fourth.endsWith("_afterif:"))) {
|
||||||
mods.add(Modification(lines[0].index, true, null))
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
mods.add(Modification(lines[1].index, true, null))
|
mods.add(Modification(lines[1].index, true, null))
|
||||||
mods.add(Modification(lines[2].index, true, null))
|
mods.add(Modification(lines[2].index, true, null))
|
||||||
}
|
}
|
||||||
else if(fourth.startsWith("label_") && (fourth.endsWith("_shortcut") || fourth.endsWith("_shortcut:"))) {
|
else if(fourth.startsWith(autoLabelPrefix) && (fourth.endsWith("_shortcut") || fourth.endsWith("_shortcut:"))) {
|
||||||
if((fifth.startsWith("beq label_") || fifth.startsWith("bne label_")) &&
|
if((fifth.startsWith("beq $autoLabelPrefix") || fifth.startsWith("bne $autoLabelPrefix")) &&
|
||||||
(fifth.endsWith("_shortcut") || fifth.endsWith("_afterif") || fifth.endsWith("_shortcut:") || fifth.endsWith("_afterif:"))) {
|
(fifth.endsWith("_shortcut") || fifth.endsWith("_afterif") || fifth.endsWith("_shortcut:") || fifth.endsWith("_afterif:"))) {
|
||||||
mods.add(Modification(lines[0].index, true, null))
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
mods.add(Modification(lines[1].index, true, null))
|
mods.add(Modification(lines[1].index, true, null))
|
||||||
@ -386,6 +393,7 @@ private fun optimizeStoreLoadSame(
|
|||||||
for (lines in linesByFour) {
|
for (lines in linesByFour) {
|
||||||
val first = lines[1].value.trimStart()
|
val first = lines[1].value.trimStart()
|
||||||
val second = lines[2].value.trimStart()
|
val second = lines[2].value.trimStart()
|
||||||
|
val third = lines[3].value.trimStart()
|
||||||
|
|
||||||
// sta X + lda X, sty X + ldy X, stx X + ldx X -> the second instruction can OFTEN be eliminated
|
// sta X + lda X, sty X + ldy X, stx X + ldx X -> the second instruction can OFTEN be eliminated
|
||||||
if ((first.startsWith("sta ") && second.startsWith("lda ")) ||
|
if ((first.startsWith("sta ") && second.startsWith("lda ")) ||
|
||||||
@ -395,7 +403,6 @@ private fun optimizeStoreLoadSame(
|
|||||||
(first.startsWith("ldy ") && second.startsWith("ldy ")) ||
|
(first.startsWith("ldy ") && second.startsWith("ldy ")) ||
|
||||||
(first.startsWith("ldx ") && second.startsWith("ldx "))
|
(first.startsWith("ldx ") && second.startsWith("ldx "))
|
||||||
) {
|
) {
|
||||||
val third = lines[3].value.trimStart()
|
|
||||||
val attemptRemove =
|
val attemptRemove =
|
||||||
if(third.isBranch()) {
|
if(third.isBranch()) {
|
||||||
// a branch instruction follows, we can only remove the load instruction if
|
// a branch instruction follows, we can only remove the load instruction if
|
||||||
@ -510,34 +517,38 @@ private fun optimizeJsrRtsAndOtherCombinations(linesByFour: Sequence<List<Indexe
|
|||||||
for (lines in linesByFour) {
|
for (lines in linesByFour) {
|
||||||
val first = lines[0].value
|
val first = lines[0].value
|
||||||
val second = lines[1].value
|
val second = lines[1].value
|
||||||
if ((" jsr" in first || "\tjsr" in first ) && (" rts" in second || "\trts" in second)) {
|
val third = lines[2].value
|
||||||
mods += Modification(lines[0].index, false, lines[0].value.replace("jsr", "jmp"))
|
|
||||||
mods += Modification(lines[1].index, true, null)
|
if(!haslabel(second)) {
|
||||||
}
|
if ((" jsr" in first || "\tjsr" in first ) && (" rts" in second || "\trts" in second)) {
|
||||||
else if (" rts" in first || "\trts" in first) {
|
if("floats.pushFAC" !in first && "floats.popFAC" !in first) { // these 2 routines depend on being called with JSR!!
|
||||||
if (" jmp" in second || "\tjmp" in second)
|
mods += Modification(lines[0].index, false, lines[0].value.replace("jsr", "jmp"))
|
||||||
mods += Modification(lines[1].index, true, null)
|
mods += Modification(lines[1].index, true, null)
|
||||||
else if (" bra" in second || "\tbra" in second)
|
}
|
||||||
mods += Modification(lines[1].index, true, null)
|
}
|
||||||
else if (" bcc" in second || "\tbcc" in second)
|
else if (" rts" in first || "\trts" in first) {
|
||||||
mods += Modification(lines[1].index, true, null)
|
if (" jmp" in second || "\tjmp" in second)
|
||||||
else if (" bcs" in second || "\tbcs" in second)
|
mods += Modification(lines[1].index, true, null)
|
||||||
mods += Modification(lines[1].index, true, null)
|
else if (" bra" in second || "\tbra" in second)
|
||||||
else if (" beq" in second || "\tbeq" in second)
|
mods += Modification(lines[1].index, true, null)
|
||||||
mods += Modification(lines[1].index, true, null)
|
else if (" bcc" in second || "\tbcc" in second)
|
||||||
else if (" bne" in second || "\tbne" in second)
|
mods += Modification(lines[1].index, true, null)
|
||||||
mods += Modification(lines[1].index, true, null)
|
else if (" bcs" in second || "\tbcs" in second)
|
||||||
else if (" bmi" in second || "\tbmi" in second)
|
mods += Modification(lines[1].index, true, null)
|
||||||
mods += Modification(lines[1].index, true, null)
|
else if (" beq" in second || "\tbeq" in second)
|
||||||
else if (" bpl" in second || "\tbpl" in second)
|
mods += Modification(lines[1].index, true, null)
|
||||||
mods += Modification(lines[1].index, true, null)
|
else if (" bne" in second || "\tbne" in second)
|
||||||
else if (" bvs" in second || "\tbvs" in second)
|
mods += Modification(lines[1].index, true, null)
|
||||||
mods += Modification(lines[1].index, true, null)
|
else if (" bmi" in second || "\tbmi" in second)
|
||||||
else if (" bvc" in second || "\tbvc" in second)
|
mods += Modification(lines[1].index, true, null)
|
||||||
mods += Modification(lines[1].index, true, null)
|
else if (" bpl" in second || "\tbpl" in second)
|
||||||
}
|
mods += Modification(lines[1].index, true, null)
|
||||||
|
else if (" bvs" in second || "\tbvs" in second)
|
||||||
|
mods += Modification(lines[1].index, true, null)
|
||||||
|
else if (" bvc" in second || "\tbvc" in second)
|
||||||
|
mods += Modification(lines[1].index, true, null)
|
||||||
|
}
|
||||||
|
|
||||||
if (!haslabel(second)) {
|
|
||||||
if ((" lda" in first || "\tlda" in first) && (" cmp #0" in second || "\tcmp #0" in second) ||
|
if ((" lda" in first || "\tlda" in first) && (" cmp #0" in second || "\tcmp #0" in second) ||
|
||||||
(" ldx" in first || "\tldx" in first) && (" cpx #0" in second || "\tcpx #0" in second) ||
|
(" ldx" in first || "\tldx" in first) && (" cpx #0" in second || "\tcpx #0" in second) ||
|
||||||
(" ldy" in first || "\tldy" in first) && (" cpy #0" in second || "\tcpy #0" in second)
|
(" ldy" in first || "\tldy" in first) && (" cpy #0" in second || "\tcpy #0" in second)
|
||||||
@ -579,6 +590,58 @@ private fun optimizeJsrRtsAndOtherCombinations(linesByFour: Sequence<List<Indexe
|
|||||||
mods += Modification(lines[3].index, true, null)
|
mods += Modification(lines[3].index, true, null)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
fun sameLabel(branchInstr: String, jumpInstr: String, labelInstr: String): Boolean {
|
||||||
|
if('(' in jumpInstr) return false // indirect jump cannot be replaced
|
||||||
|
val label = labelInstr.trimEnd().substringBefore(':').substringBefore(' ').substringBefore('\t')
|
||||||
|
val branchLabel = branchInstr.trimStart().substring(3).trim()
|
||||||
|
return label==branchLabel
|
||||||
|
}
|
||||||
|
|
||||||
|
// beq Label + jmp Addr + Label -> bne Addr
|
||||||
|
if((" jmp" in second || "\tjmp " in second) && haslabel(third)) {
|
||||||
|
if((" beq " in first || "\tbeq " in first) && sameLabel(first, second, third)) {
|
||||||
|
val branch = second.replace("jmp", "bne")
|
||||||
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, branch))
|
||||||
|
}
|
||||||
|
else if((" bne " in first || "\tbne " in first) && sameLabel(first, second, third)) {
|
||||||
|
val branch = second.replace("jmp", "beq")
|
||||||
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, branch))
|
||||||
|
}
|
||||||
|
else if((" bcc " in first || "\tbcc " in first) && sameLabel(first, second, third)){
|
||||||
|
val branch = second.replace("jmp", "bcs")
|
||||||
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, branch))
|
||||||
|
}
|
||||||
|
else if((" bcs " in first || "\tbcs " in first) && sameLabel(first, second, third)) {
|
||||||
|
val branch = second.replace("jmp", "bcc")
|
||||||
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, branch))
|
||||||
|
}
|
||||||
|
else if((" bpl " in first || "\tbpl " in first) && sameLabel(first, second, third)) {
|
||||||
|
val branch = second.replace("jmp", "bmi")
|
||||||
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, branch))
|
||||||
|
}
|
||||||
|
else if((" bmi " in first || "\tbmi " in first) && sameLabel(first, second, third)) {
|
||||||
|
val branch = second.replace("jmp", "bpl")
|
||||||
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, branch))
|
||||||
|
}
|
||||||
|
else if((" bvc " in first || "\tbvc " in first) && sameLabel(first, second, third)) {
|
||||||
|
val branch = second.replace("jmp", "bvs")
|
||||||
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, branch))
|
||||||
|
}
|
||||||
|
else if((" bvs " in first || "\tbvs " in first) && sameLabel(first, second, third)) {
|
||||||
|
val branch = second.replace("jmp", "bvc")
|
||||||
|
mods.add(Modification(lines[0].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, branch))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return mods
|
return mods
|
||||||
}
|
}
|
||||||
@ -616,6 +679,52 @@ private fun optimizeUselessPushPopStack(linesByFour: Sequence<List<IndexedValue<
|
|||||||
optimize('a', lines)
|
optimize('a', lines)
|
||||||
optimize('x', lines)
|
optimize('x', lines)
|
||||||
optimize('y', lines)
|
optimize('y', lines)
|
||||||
|
|
||||||
|
val first = lines[1].value.trimStart()
|
||||||
|
val second = lines[2].value.trimStart()
|
||||||
|
val third = lines[3].value.trimStart()
|
||||||
|
|
||||||
|
// phy + ldy + pla -> tya + ldy
|
||||||
|
// phx + ldx + pla -> txa + ldx
|
||||||
|
// pha + lda + pla -> nop
|
||||||
|
if(first=="phy" && second.startsWith("ldy ") && third=="pla") {
|
||||||
|
mods.add(Modification(lines[3].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, " tya"))
|
||||||
|
}
|
||||||
|
else if(first=="phx" && second.startsWith("ldx ") && third=="pla") {
|
||||||
|
mods.add(Modification(lines[3].index, true, null))
|
||||||
|
mods.add(Modification(lines[1].index, false, " txa"))
|
||||||
|
}
|
||||||
|
else if(first=="pha" && second.startsWith("lda ") && third=="pla") {
|
||||||
|
mods.add(Modification(lines[1].index, true, null))
|
||||||
|
mods.add(Modification(lines[2].index, true, null))
|
||||||
|
mods.add(Modification(lines[3].index, true, null))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
return mods
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private fun optimizeTSBtoRegularOr(linesByFour: Sequence<List<IndexedValue<String>>>): List<Modification> {
|
||||||
|
// Asm peephole: lda var2 / tsb var1 / lda var1 Replace this with this to save 1 cycle: lda var1 / ora var2 / sta var1
|
||||||
|
val mods = mutableListOf<Modification>()
|
||||||
|
|
||||||
|
for(lines in linesByFour) {
|
||||||
|
val first = lines[0].value.trimStart()
|
||||||
|
val second = lines[1].value.trimStart()
|
||||||
|
val third = lines[2].value.trimStart()
|
||||||
|
if(first.startsWith("lda") && second.startsWith("tsb") && third.startsWith("lda")) {
|
||||||
|
val operand1 = first.substring(3)
|
||||||
|
val operand2 = second.substring(3)
|
||||||
|
val operand3 = third.substring(3)
|
||||||
|
if(operand1!=operand2 && operand2==operand3) {
|
||||||
|
mods.add(Modification(lines[0].index, false, " lda $operand2"))
|
||||||
|
mods.add(Modification(lines[1].index, false, " ora $operand1"))
|
||||||
|
mods.add(Modification(lines[2].index, false, " sta $operand2"))
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return mods
|
return mods
|
||||||
}
|
}
|
||||||
|
@ -11,18 +11,22 @@ fun asmsub6502ArgsEvalOrder(sub: PtAsmSub): List<Int> {
|
|||||||
// 1) cx16 virtual word registers,
|
// 1) cx16 virtual word registers,
|
||||||
// 2) paired CPU registers,
|
// 2) paired CPU registers,
|
||||||
// 3) single CPU registers (order Y,X,A),
|
// 3) single CPU registers (order Y,X,A),
|
||||||
// 4) CPU Carry status flag
|
// 4) floating point registers (FAC1, FAC2),
|
||||||
|
// 5) CPU Carry status flag
|
||||||
val args = sub.parameters.withIndex()
|
val args = sub.parameters.withIndex()
|
||||||
val (cx16regs, args2) = args.partition { it.value.first.registerOrPair in Cx16VirtualRegisters }
|
val (cx16regs, args2) = args.partition { it.value.first.registerOrPair in Cx16VirtualRegisters }
|
||||||
val pairedRegisters = arrayOf(RegisterOrPair.AX, RegisterOrPair.AY, RegisterOrPair.XY)
|
val pairedRegisters = arrayOf(RegisterOrPair.AX, RegisterOrPair.AY, RegisterOrPair.XY)
|
||||||
val (pairedRegs , args3) = args2.partition { it.value.first.registerOrPair in pairedRegisters }
|
val (pairedRegs , args3) = args2.partition { it.value.first.registerOrPair in pairedRegisters }
|
||||||
val (singleRegs, rest) = args3.partition { it.value.first.registerOrPair != null }
|
val (singleRegsMixed, rest) = args3.partition { it.value.first.registerOrPair != null }
|
||||||
|
val (singleCpuRegs, floatRegs) = singleRegsMixed.partition {it.value.first.registerOrPair != RegisterOrPair.FAC1 && it.value.first.registerOrPair != RegisterOrPair.FAC2 }
|
||||||
|
|
||||||
cx16regs.forEach { order += it.index }
|
cx16regs.forEach { order += it.index }
|
||||||
pairedRegs.forEach { order += it.index }
|
pairedRegs.forEach { order += it.index }
|
||||||
singleRegs.sortedBy { it.value.first.registerOrPair!!.asCpuRegister() }.asReversed().forEach { order += it.index }
|
singleCpuRegs.sortedBy { it.value.first.registerOrPair!!.asCpuRegister() }.asReversed().forEach { order += it.index }
|
||||||
require(rest.all { it.value.first.registerOrPair==null && it.value.first.statusflag!=null})
|
require(rest.all { it.value.first.registerOrPair==null && it.value.first.statusflag!=null})
|
||||||
|
floatRegs.forEach { order += it.index }
|
||||||
rest.forEach { order += it.index }
|
rest.forEach { order += it.index }
|
||||||
require(order.size==sub.parameters.size)
|
require(order.size==sub.parameters.size)
|
||||||
|
|
||||||
return order
|
return order
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,10 @@
|
|||||||
package prog8.codegen.cpu6502
|
package prog8.codegen.cpu6502
|
||||||
|
|
||||||
|
import prog8.code.ast.PtLabel
|
||||||
import prog8.code.core.*
|
import prog8.code.core.*
|
||||||
|
import prog8.code.target.AtariTarget
|
||||||
import prog8.code.target.C64Target
|
import prog8.code.target.C64Target
|
||||||
|
import prog8.code.target.Neo6502Target
|
||||||
import java.nio.file.Path
|
import java.nio.file.Path
|
||||||
|
|
||||||
|
|
||||||
@ -16,19 +19,20 @@ internal class AssemblyProgram(
|
|||||||
private val binFile = outputDir.resolve("$name.bin")
|
private val binFile = outputDir.resolve("$name.bin")
|
||||||
private val viceMonListFile = outputDir.resolve(C64Target.viceMonListName(name))
|
private val viceMonListFile = outputDir.resolve(C64Target.viceMonListName(name))
|
||||||
private val listFile = outputDir.resolve("$name.list")
|
private val listFile = outputDir.resolve("$name.list")
|
||||||
|
private val targetWithoutBreakpointsForEmulator = arrayOf(AtariTarget.NAME, Neo6502Target.NAME)
|
||||||
|
|
||||||
override fun assemble(options: CompilationOptions, errors: IErrorReporter): Boolean {
|
override fun assemble(options: CompilationOptions, errors: IErrorReporter): Boolean {
|
||||||
|
|
||||||
val assemblerCommand: List<String>
|
val assemblerCommand: List<String>
|
||||||
|
|
||||||
when (compTarget.name) {
|
when (compTarget.name) {
|
||||||
in setOf("c64", "c128", "cx16", "pet32") -> {
|
in arrayOf("c64", "c128", "cx16", "pet32") -> {
|
||||||
// CBM machines .prg generation.
|
// CBM machines .prg generation.
|
||||||
|
|
||||||
// add "-Wlong-branch" to see warnings about conversion of branch instructions to jumps (default = do this silently)
|
// add "-Wlong-branch" to see warnings about conversion of branch instructions to jumps (default = do this silently)
|
||||||
val command = mutableListOf("64tass", "--ascii", "--case-sensitive", "--long-branch",
|
val command = mutableListOf("64tass", "--ascii", "--case-sensitive", "--long-branch",
|
||||||
"-Wall", // "-Wno-strict-bool", "-Werror",
|
"-Wall", // "-Wno-strict-bool", "-Werror",
|
||||||
"--dump-labels", "--vice-labels", "--labels=$viceMonListFile", "--no-monitor"
|
"--dump-labels", "--vice-labels", "--labels=$viceMonListFile"
|
||||||
)
|
)
|
||||||
|
|
||||||
if(options.warnSymbolShadowing)
|
if(options.warnSymbolShadowing)
|
||||||
@ -39,8 +43,9 @@ internal class AssemblyProgram(
|
|||||||
if(options.asmQuiet)
|
if(options.asmQuiet)
|
||||||
command.add("--quiet")
|
command.add("--quiet")
|
||||||
|
|
||||||
if(options.asmListfile)
|
if(options.asmListfile) {
|
||||||
command.add("--list=$listFile")
|
command.addAll(listOf("--list=$listFile", "--no-monitor"))
|
||||||
|
}
|
||||||
|
|
||||||
val outFile = when (options.output) {
|
val outFile = when (options.output) {
|
||||||
OutputType.PRG -> {
|
OutputType.PRG -> {
|
||||||
@ -95,12 +100,47 @@ internal class AssemblyProgram(
|
|||||||
command.addAll(listOf("--output", outFile.toString(), assemblyFile.toString()))
|
command.addAll(listOf("--output", outFile.toString(), assemblyFile.toString()))
|
||||||
assemblerCommand = command
|
assemblerCommand = command
|
||||||
}
|
}
|
||||||
|
"neo" -> {
|
||||||
|
// Neo6502 raw program generation.
|
||||||
|
|
||||||
|
if(options.output!=OutputType.RAW || options.loadAddress!=0x0800u || options.launcher!=CbmPrgLauncherType.NONE) {
|
||||||
|
throw AssemblyError("invalid program compilation options. Neo6502 requires %output raw, %launcher none, %address $0800")
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO are these options okay for neo?
|
||||||
|
val command = mutableListOf("64tass", "--case-sensitive", "--long-branch",
|
||||||
|
"-Wall", // "-Werror", "-Wno-strict-bool"
|
||||||
|
"--no-monitor"
|
||||||
|
)
|
||||||
|
|
||||||
|
if(options.warnSymbolShadowing)
|
||||||
|
command.add("-Wshadow")
|
||||||
|
else
|
||||||
|
command.add("-Wno-shadow")
|
||||||
|
|
||||||
|
if(options.asmQuiet)
|
||||||
|
command.add("--quiet")
|
||||||
|
|
||||||
|
if(options.asmListfile)
|
||||||
|
command.add("--list=$listFile")
|
||||||
|
|
||||||
|
val outFile = when (options.output) {
|
||||||
|
OutputType.RAW -> {
|
||||||
|
command.add("--nostart")
|
||||||
|
println("\nCreating raw binary for target ${compTarget.name}.")
|
||||||
|
binFile
|
||||||
|
}
|
||||||
|
else -> throw AssemblyError("invalid output type, need 'raw'")
|
||||||
|
}
|
||||||
|
command.addAll(listOf("--output", outFile.toString(), assemblyFile.toString()))
|
||||||
|
assemblerCommand = command
|
||||||
|
}
|
||||||
else -> throw AssemblyError("invalid compilation target")
|
else -> throw AssemblyError("invalid compilation target")
|
||||||
}
|
}
|
||||||
|
|
||||||
val proc = ProcessBuilder(assemblerCommand).inheritIO().start()
|
val proc = ProcessBuilder(assemblerCommand).inheritIO().start()
|
||||||
val result = proc.waitFor()
|
val result = proc.waitFor()
|
||||||
if (result == 0 && compTarget.name!="atari") {
|
if (result == 0 && compTarget.name !in targetWithoutBreakpointsForEmulator) {
|
||||||
removeGeneratedLabelsFromMonlist()
|
removeGeneratedLabelsFromMonlist()
|
||||||
generateBreakpointList()
|
generateBreakpointList()
|
||||||
}
|
}
|
||||||
@ -108,7 +148,7 @@ internal class AssemblyProgram(
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun removeGeneratedLabelsFromMonlist() {
|
private fun removeGeneratedLabelsFromMonlist() {
|
||||||
val pattern = Regex("""al (\w+) \S+prog8_label_.+?""")
|
val pattern = Regex("""al (\w+) \S+${PtLabel.GENERATED_LABEL_PREFIX}.+?""")
|
||||||
val lines = viceMonListFile.toFile().readLines()
|
val lines = viceMonListFile.toFile().readLines()
|
||||||
viceMonListFile.toFile().outputStream().bufferedWriter().use {
|
viceMonListFile.toFile().outputStream().bufferedWriter().use {
|
||||||
for (line in lines) {
|
for (line in lines) {
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -3,7 +3,9 @@ package prog8.codegen.cpu6502
|
|||||||
import prog8.code.ast.IPtSubroutine
|
import prog8.code.ast.IPtSubroutine
|
||||||
import prog8.code.ast.PtAsmSub
|
import prog8.code.ast.PtAsmSub
|
||||||
import prog8.code.ast.PtSub
|
import prog8.code.ast.PtSub
|
||||||
import prog8.code.core.*
|
import prog8.code.core.DataType
|
||||||
|
import prog8.code.core.RegisterOrPair
|
||||||
|
import prog8.code.core.RegisterOrStatusflag
|
||||||
|
|
||||||
|
|
||||||
internal fun IPtSubroutine.returnsWhatWhere(): List<Pair<RegisterOrStatusflag, DataType>> {
|
internal fun IPtSubroutine.returnsWhatWhere(): List<Pair<RegisterOrStatusflag, DataType>> {
|
||||||
@ -16,10 +18,10 @@ internal fun IPtSubroutine.returnsWhatWhere(): List<Pair<RegisterOrStatusflag, D
|
|||||||
return if(returntype==null)
|
return if(returntype==null)
|
||||||
emptyList()
|
emptyList()
|
||||||
else {
|
else {
|
||||||
val register = when (returntype!!) {
|
val register = when {
|
||||||
in ByteDatatypesWithBoolean -> RegisterOrStatusflag(RegisterOrPair.A, null)
|
returntype!!.isByteOrBool -> RegisterOrStatusflag(RegisterOrPair.A, null)
|
||||||
in WordDatatypes -> RegisterOrStatusflag(RegisterOrPair.AY, null)
|
returntype!!.isWord -> RegisterOrStatusflag(RegisterOrPair.AY, null)
|
||||||
DataType.FLOAT -> RegisterOrStatusflag(RegisterOrPair.FAC1, null)
|
returntype!!.isFloat -> RegisterOrStatusflag(RegisterOrPair.FAC1, null)
|
||||||
else -> RegisterOrStatusflag(RegisterOrPair.AY, null)
|
else -> RegisterOrStatusflag(RegisterOrPair.AY, null)
|
||||||
}
|
}
|
||||||
listOf(Pair(register, returntype!!))
|
listOf(Pair(register, returntype!!))
|
||||||
@ -30,11 +32,11 @@ internal fun IPtSubroutine.returnsWhatWhere(): List<Pair<RegisterOrStatusflag, D
|
|||||||
|
|
||||||
|
|
||||||
internal fun PtSub.returnRegister(): RegisterOrStatusflag? {
|
internal fun PtSub.returnRegister(): RegisterOrStatusflag? {
|
||||||
return when(returntype) {
|
return when {
|
||||||
in ByteDatatypesWithBoolean -> RegisterOrStatusflag(RegisterOrPair.A, null)
|
returntype?.isByteOrBool==true -> RegisterOrStatusflag(RegisterOrPair.A, null)
|
||||||
in WordDatatypes -> RegisterOrStatusflag(RegisterOrPair.AY, null)
|
returntype?.isWord==true -> RegisterOrStatusflag(RegisterOrPair.AY, null)
|
||||||
DataType.FLOAT -> RegisterOrStatusflag(RegisterOrPair.FAC1, null)
|
returntype?.isFloat==true -> RegisterOrStatusflag(RegisterOrPair.FAC1, null)
|
||||||
null -> null
|
returntype==null -> null
|
||||||
else -> RegisterOrStatusflag(RegisterOrPair.AY, null)
|
else -> RegisterOrStatusflag(RegisterOrPair.AY, null)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -37,7 +37,7 @@ internal class ForLoopsAsmGen(
|
|||||||
val endLabel = asmgen.makeLabel("for_end")
|
val endLabel = asmgen.makeLabel("for_end")
|
||||||
val modifiedLabel = asmgen.makeLabel("for_modified")
|
val modifiedLabel = asmgen.makeLabel("for_modified")
|
||||||
val modifiedLabel2 = asmgen.makeLabel("for_modifiedb")
|
val modifiedLabel2 = asmgen.makeLabel("for_modifiedb")
|
||||||
asmgen.loopEndLabels.push(endLabel)
|
asmgen.loopEndLabels.add(endLabel)
|
||||||
val stepsize=range.step.asConstInteger()!!
|
val stepsize=range.step.asConstInteger()!!
|
||||||
|
|
||||||
if(stepsize < -1) {
|
if(stepsize < -1) {
|
||||||
@ -46,27 +46,45 @@ internal class ForLoopsAsmGen(
|
|||||||
throw AssemblyError("for unsigned loop variable it's not possible to count down with step != -1 from a non-const value to exactly zero due to value wrapping")
|
throw AssemblyError("for unsigned loop variable it's not possible to count down with step != -1 from a non-const value to exactly zero due to value wrapping")
|
||||||
}
|
}
|
||||||
|
|
||||||
when(iterableDt) {
|
when {
|
||||||
DataType.ARRAY_B, DataType.ARRAY_UB -> {
|
iterableDt.isByteArray -> {
|
||||||
if (stepsize==1 || stepsize==-1) {
|
val varname = asmgen.asmVariableName(stmt.variable)
|
||||||
|
asmgen.assignExpressionToVariable(range.from, varname, iterableDt.elementType())
|
||||||
|
if (stepsize==-1 && range.to.asConstInteger()==0) {
|
||||||
|
// simple loop downto 0 step -1
|
||||||
|
asmgen.out(loopLabel)
|
||||||
|
asmgen.translate(stmt.statements)
|
||||||
|
asmgen.out("""
|
||||||
|
dec $varname
|
||||||
|
lda $varname
|
||||||
|
cmp #255
|
||||||
|
bne $loopLabel""")
|
||||||
|
}
|
||||||
|
else if (stepsize==-1 && range.to.asConstInteger()==1) {
|
||||||
|
// simple loop downto 1 step -1
|
||||||
|
asmgen.out(loopLabel)
|
||||||
|
asmgen.translate(stmt.statements)
|
||||||
|
asmgen.out("""
|
||||||
|
dec $varname
|
||||||
|
bne $loopLabel""")
|
||||||
|
}
|
||||||
|
else if (stepsize==1 || stepsize==-1) {
|
||||||
// bytes array, step 1 or -1
|
// bytes array, step 1 or -1
|
||||||
|
|
||||||
val incdec = if(stepsize==1) "inc" else "dec"
|
val incdec = if(stepsize==1) "inc" else "dec"
|
||||||
// loop over byte range via loopvar
|
// loop over byte range via loopvar
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
|
||||||
asmgen.assignExpressionToVariable(range.from, varname, ArrayToElementTypes.getValue(iterableDt))
|
|
||||||
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.A, false)
|
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.A, false)
|
||||||
// pre-check for end already reached
|
// pre-check for end already reached
|
||||||
if(iterableDt==DataType.ARRAY_B) {
|
if(iterableDt.isSignedByteArray) {
|
||||||
asmgen.out(" sta $modifiedLabel+1")
|
asmgen.out(" sta $modifiedLabel+1")
|
||||||
if(stepsize<0)
|
if(stepsize<0) {
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
clc
|
clc
|
||||||
sbc $varname
|
sbc $varname
|
||||||
bvc +
|
bvc +
|
||||||
eor #${'$'}80
|
eor #${'$'}80
|
||||||
+ bpl $endLabel""")
|
+ bpl $endLabel""")
|
||||||
|
}
|
||||||
else
|
else
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
sec
|
sec
|
||||||
@ -75,12 +93,13 @@ internal class ForLoopsAsmGen(
|
|||||||
eor #${'$'}80
|
eor #${'$'}80
|
||||||
+ bmi $endLabel""")
|
+ bmi $endLabel""")
|
||||||
} else {
|
} else {
|
||||||
if(stepsize<0)
|
if(stepsize<0) {
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
cmp $varname
|
cmp $varname
|
||||||
beq +
|
beq +
|
||||||
bcs $endLabel
|
bcs $endLabel
|
||||||
+""")
|
+""")
|
||||||
|
}
|
||||||
else
|
else
|
||||||
asmgen.out(" cmp $varname | bcc $endLabel")
|
asmgen.out(" cmp $varname | bcc $endLabel")
|
||||||
asmgen.out(" sta $modifiedLabel+1")
|
asmgen.out(" sta $modifiedLabel+1")
|
||||||
@ -100,11 +119,9 @@ $modifiedLabel cmp #0 ; modified
|
|||||||
// bytes, step >= 2 or <= -2
|
// bytes, step >= 2 or <= -2
|
||||||
|
|
||||||
// loop over byte range via loopvar
|
// loop over byte range via loopvar
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
|
||||||
asmgen.assignExpressionToVariable(range.from, varname, ArrayToElementTypes.getValue(iterableDt))
|
|
||||||
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.A, false)
|
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.A, false)
|
||||||
// pre-check for end already reached
|
// pre-check for end already reached
|
||||||
if(iterableDt==DataType.ARRAY_B) {
|
if(iterableDt.isSignedByteArray) {
|
||||||
asmgen.out(" sta $modifiedLabel+1")
|
asmgen.out(" sta $modifiedLabel+1")
|
||||||
if(stepsize<0)
|
if(stepsize<0)
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
@ -154,59 +171,86 @@ $modifiedLabel cmp #0 ; modified
|
|||||||
asmgen.out(endLabel)
|
asmgen.out(endLabel)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DataType.ARRAY_W, DataType.ARRAY_UW -> {
|
iterableDt.isWordArray && !iterableDt.isSplitWordArray -> {
|
||||||
when {
|
val varname = asmgen.asmVariableName(stmt.variable)
|
||||||
|
assignLoopvarWord(stmt, range)
|
||||||
|
if(stepsize==-1 && range.to.asConstInteger()==0) {
|
||||||
|
// simple loop downto 0 step -1 (words)
|
||||||
|
asmgen.out(loopLabel)
|
||||||
|
asmgen.translate(stmt.statements)
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varname
|
||||||
|
bne ++
|
||||||
|
lda $varname+1
|
||||||
|
beq $endLabel
|
||||||
|
+ lda $varname
|
||||||
|
bne +
|
||||||
|
dec $varname+1
|
||||||
|
+ dec $varname""")
|
||||||
|
asmgen.jmp(loopLabel)
|
||||||
|
asmgen.out(endLabel)
|
||||||
|
}
|
||||||
|
else if (stepsize==-1 && range.to.asConstInteger()==1) {
|
||||||
|
// simple loop downto 1 step -1 (words)
|
||||||
|
asmgen.out(loopLabel)
|
||||||
|
asmgen.translate(stmt.statements)
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varname
|
||||||
|
cmp #1
|
||||||
|
bne +
|
||||||
|
lda $varname+1
|
||||||
|
beq $endLabel
|
||||||
|
+ lda $varname
|
||||||
|
bne +
|
||||||
|
dec $varname+1
|
||||||
|
+ dec $varname""")
|
||||||
|
asmgen.jmp(loopLabel)
|
||||||
|
asmgen.out(endLabel)
|
||||||
|
}
|
||||||
|
else if (stepsize == 1 || stepsize == -1) {
|
||||||
// words, step 1 or -1
|
// words, step 1 or -1
|
||||||
|
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY)
|
||||||
stepsize == 1 || stepsize == -1 -> {
|
precheckFromToWord(iterableDt, stepsize, varname, endLabel)
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
asmgen.out("""
|
||||||
assignLoopvarWord(stmt, range)
|
sty $modifiedLabel+1
|
||||||
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY)
|
sta $modifiedLabel2+1
|
||||||
precheckFromToWord(iterableDt, stepsize, varname, endLabel)
|
|
||||||
asmgen.out("""
|
|
||||||
sty $modifiedLabel+1
|
|
||||||
sta $modifiedLabel2+1
|
|
||||||
$loopLabel""")
|
$loopLabel""")
|
||||||
asmgen.translate(stmt.statements)
|
asmgen.translate(stmt.statements)
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda $varname+1
|
lda $varname+1
|
||||||
$modifiedLabel cmp #0 ; modified
|
$modifiedLabel cmp #0 ; modified
|
||||||
bne +
|
bne +
|
||||||
lda $varname
|
lda $varname
|
||||||
$modifiedLabel2 cmp #0 ; modified
|
$modifiedLabel2 cmp #0 ; modified
|
||||||
beq $endLabel""")
|
beq $endLabel""")
|
||||||
if(stepsize==1) {
|
if(stepsize==1) {
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
+ inc $varname
|
+ inc $varname
|
||||||
bne $loopLabel
|
bne $loopLabel
|
||||||
inc $varname+1""")
|
inc $varname+1""")
|
||||||
asmgen.jmp(loopLabel)
|
asmgen.jmp(loopLabel)
|
||||||
} else {
|
} else {
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
+ lda $varname
|
+ lda $varname
|
||||||
bne +
|
bne +
|
||||||
dec $varname+1
|
dec $varname+1
|
||||||
+ dec $varname""")
|
+ dec $varname""")
|
||||||
asmgen.jmp(loopLabel)
|
asmgen.jmp(loopLabel)
|
||||||
}
|
|
||||||
asmgen.out(endLabel)
|
|
||||||
}
|
}
|
||||||
stepsize > 0 -> {
|
asmgen.out(endLabel)
|
||||||
|
}
|
||||||
// (u)words, step >= 2
|
else if (stepsize > 0) {
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
// (u)words, step >= 2
|
||||||
assignLoopvarWord(stmt, range)
|
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY)
|
||||||
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY)
|
precheckFromToWord(iterableDt, stepsize, varname, endLabel)
|
||||||
precheckFromToWord(iterableDt, stepsize, varname, endLabel)
|
asmgen.out("""
|
||||||
asmgen.out("""
|
sty $modifiedLabel+1
|
||||||
sty $modifiedLabel+1
|
sta $modifiedLabel2+1
|
||||||
sta $modifiedLabel2+1
|
|
||||||
$loopLabel""")
|
$loopLabel""")
|
||||||
asmgen.translate(stmt.statements)
|
asmgen.translate(stmt.statements)
|
||||||
|
|
||||||
if (iterableDt == DataType.ARRAY_UW) {
|
if (iterableDt.isUnsignedWordArray) {
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda $varname
|
lda $varname
|
||||||
clc
|
clc
|
||||||
adc #<$stepsize
|
adc #<$stepsize
|
||||||
@ -222,8 +266,8 @@ $modifiedLabel2 lda #0 ; modified
|
|||||||
bcc $endLabel
|
bcc $endLabel
|
||||||
bcs $loopLabel
|
bcs $loopLabel
|
||||||
$endLabel""")
|
$endLabel""")
|
||||||
} else {
|
} else {
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda $varname
|
lda $varname
|
||||||
clc
|
clc
|
||||||
adc #<$stepsize
|
adc #<$stepsize
|
||||||
@ -239,22 +283,20 @@ $modifiedLabel lda #0 ; modified
|
|||||||
eor #$80
|
eor #$80
|
||||||
+ bpl $loopLabel
|
+ bpl $loopLabel
|
||||||
$endLabel""")
|
$endLabel""")
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else -> {
|
}
|
||||||
|
else {
|
||||||
|
|
||||||
// (u)words, step <= -2
|
// (u)words, step <= -2
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY)
|
||||||
assignLoopvarWord(stmt, range)
|
precheckFromToWord(iterableDt, stepsize, varname, endLabel)
|
||||||
asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY)
|
asmgen.out("""
|
||||||
precheckFromToWord(iterableDt, stepsize, varname, endLabel)
|
sty $modifiedLabel+1
|
||||||
asmgen.out("""
|
sta $modifiedLabel2+1
|
||||||
sty $modifiedLabel+1
|
|
||||||
sta $modifiedLabel2+1
|
|
||||||
$loopLabel""")
|
$loopLabel""")
|
||||||
asmgen.translate(stmt.statements)
|
asmgen.translate(stmt.statements)
|
||||||
|
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda $varname
|
lda $varname
|
||||||
sec
|
sec
|
||||||
sbc #<${stepsize.absoluteValue}
|
sbc #<${stepsize.absoluteValue}
|
||||||
@ -271,19 +313,18 @@ $modifiedLabel sbc #0 ; modified
|
|||||||
eor #$80
|
eor #$80
|
||||||
+ bpl $loopLabel
|
+ bpl $loopLabel
|
||||||
$endLabel""")
|
$endLabel""")
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else -> throw AssemblyError("range expression can only be byte or word")
|
else -> throw AssemblyError("range expression can only be byte or word")
|
||||||
}
|
}
|
||||||
|
|
||||||
asmgen.loopEndLabels.pop()
|
asmgen.loopEndLabels.removeLast()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun precheckFromToWord(iterableDt: DataType, stepsize: Int, fromVar: String, endLabel: String) {
|
private fun precheckFromToWord(iterableDt: DataType, stepsize: Int, fromVar: String, endLabel: String) {
|
||||||
// pre-check for end already reached.
|
// pre-check for end already reached.
|
||||||
// 'to' is in AY, do NOT clobber this!
|
// 'to' is in AY, do NOT clobber this!
|
||||||
if(iterableDt==DataType.ARRAY_W) {
|
if(iterableDt.isSignedWordArray) {
|
||||||
if(stepsize<0)
|
if(stepsize<0)
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
sta P8ZP_SCRATCH_W2 ; to
|
sta P8ZP_SCRATCH_W2 ; to
|
||||||
@ -333,16 +374,15 @@ $endLabel""")
|
|||||||
private fun translateForOverIterableVar(stmt: PtForLoop, iterableDt: DataType, ident: PtIdentifier) {
|
private fun translateForOverIterableVar(stmt: PtForLoop, iterableDt: DataType, ident: PtIdentifier) {
|
||||||
val loopLabel = asmgen.makeLabel("for_loop")
|
val loopLabel = asmgen.makeLabel("for_loop")
|
||||||
val endLabel = asmgen.makeLabel("for_end")
|
val endLabel = asmgen.makeLabel("for_end")
|
||||||
asmgen.loopEndLabels.push(endLabel)
|
asmgen.loopEndLabels.add(endLabel)
|
||||||
val iterableName = asmgen.asmVariableName(ident)
|
val iterableName = asmgen.asmVariableName(ident)
|
||||||
val symbol = asmgen.symbolTable.lookup(ident.name)
|
val numElements = when(val symbol = asmgen.symbolTable.lookup(ident.name)) {
|
||||||
val numElements = when(symbol) {
|
|
||||||
is StStaticVariable -> symbol.length!!
|
is StStaticVariable -> symbol.length!!
|
||||||
is StMemVar -> symbol.length!!
|
is StMemVar -> symbol.length!!
|
||||||
else -> 0
|
else -> 0
|
||||||
}
|
}
|
||||||
when(iterableDt) {
|
when {
|
||||||
DataType.STR -> {
|
iterableDt.isString -> {
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda #<$iterableName
|
lda #<$iterableName
|
||||||
ldy #>$iterableName
|
ldy #>$iterableName
|
||||||
@ -359,7 +399,7 @@ $loopLabel lda ${65535.toHex()} ; modified
|
|||||||
bne $loopLabel
|
bne $loopLabel
|
||||||
$endLabel""")
|
$endLabel""")
|
||||||
}
|
}
|
||||||
DataType.ARRAY_UB, DataType.ARRAY_B, DataType.ARRAY_BOOL -> {
|
iterableDt.isByteArray || iterableDt.isBoolArray -> {
|
||||||
val indexVar = asmgen.makeLabel("for_index")
|
val indexVar = asmgen.makeLabel("for_index")
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
ldy #0
|
ldy #0
|
||||||
@ -384,7 +424,7 @@ $loopLabel sty $indexVar
|
|||||||
}
|
}
|
||||||
if(numElements>=16) {
|
if(numElements>=16) {
|
||||||
// allocate index var on ZP if possible
|
// allocate index var on ZP if possible
|
||||||
val result = zeropage.allocate(indexVar, DataType.UBYTE, null, stmt.position, asmgen.errors)
|
val result = zeropage.allocate(indexVar, DataType.forDt(BaseDataType.UBYTE), null, stmt.position, asmgen.errors)
|
||||||
result.fold(
|
result.fold(
|
||||||
success = { (address, _, _)-> asmgen.out("""$indexVar = $address ; auto zp UBYTE""") },
|
success = { (address, _, _)-> asmgen.out("""$indexVar = $address ; auto zp UBYTE""") },
|
||||||
failure = { asmgen.out("$indexVar .byte 0") }
|
failure = { asmgen.out("$indexVar .byte 0") }
|
||||||
@ -394,7 +434,45 @@ $loopLabel sty $indexVar
|
|||||||
}
|
}
|
||||||
asmgen.out(endLabel)
|
asmgen.out(endLabel)
|
||||||
}
|
}
|
||||||
DataType.ARRAY_W, DataType.ARRAY_UW -> {
|
iterableDt.isSplitWordArray -> {
|
||||||
|
val indexVar = asmgen.makeLabel("for_index")
|
||||||
|
val loopvarName = asmgen.asmVariableName(stmt.variable)
|
||||||
|
asmgen.out("""
|
||||||
|
ldy #0
|
||||||
|
$loopLabel sty $indexVar
|
||||||
|
lda ${iterableName}_lsb,y
|
||||||
|
sta $loopvarName
|
||||||
|
lda ${iterableName}_msb,y
|
||||||
|
sta $loopvarName+1""")
|
||||||
|
asmgen.translate(stmt.statements)
|
||||||
|
if(numElements<=255) {
|
||||||
|
asmgen.out("""
|
||||||
|
ldy $indexVar
|
||||||
|
iny
|
||||||
|
cpy #$numElements
|
||||||
|
beq $endLabel
|
||||||
|
bne $loopLabel""")
|
||||||
|
} else {
|
||||||
|
// length is 256
|
||||||
|
asmgen.out("""
|
||||||
|
ldy $indexVar
|
||||||
|
iny
|
||||||
|
bne $loopLabel
|
||||||
|
beq $endLabel""")
|
||||||
|
}
|
||||||
|
if(numElements>=16) {
|
||||||
|
// allocate index var on ZP if possible
|
||||||
|
val result = zeropage.allocate(indexVar, DataType.forDt(BaseDataType.UBYTE), null, stmt.position, asmgen.errors)
|
||||||
|
result.fold(
|
||||||
|
success = { (address,_,_)-> asmgen.out("""$indexVar = $address ; auto zp UBYTE""") },
|
||||||
|
failure = { asmgen.out("$indexVar .byte 0") }
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
asmgen.out("$indexVar .byte 0")
|
||||||
|
}
|
||||||
|
asmgen.out(endLabel)
|
||||||
|
}
|
||||||
|
iterableDt.isWordArray -> {
|
||||||
val length = numElements * 2
|
val length = numElements * 2
|
||||||
val indexVar = asmgen.makeLabel("for_index")
|
val indexVar = asmgen.makeLabel("for_index")
|
||||||
val loopvarName = asmgen.asmVariableName(stmt.variable)
|
val loopvarName = asmgen.asmVariableName(stmt.variable)
|
||||||
@ -425,7 +503,7 @@ $loopLabel sty $indexVar
|
|||||||
}
|
}
|
||||||
if(length>=16) {
|
if(length>=16) {
|
||||||
// allocate index var on ZP if possible
|
// allocate index var on ZP if possible
|
||||||
val result = zeropage.allocate(indexVar, DataType.UBYTE, null, stmt.position, asmgen.errors)
|
val result = zeropage.allocate(indexVar, DataType.forDt(BaseDataType.UBYTE), null, stmt.position, asmgen.errors)
|
||||||
result.fold(
|
result.fold(
|
||||||
success = { (address,_,_)-> asmgen.out("""$indexVar = $address ; auto zp UBYTE""") },
|
success = { (address,_,_)-> asmgen.out("""$indexVar = $address ; auto zp UBYTE""") },
|
||||||
failure = { asmgen.out("$indexVar .byte 0") }
|
failure = { asmgen.out("$indexVar .byte 0") }
|
||||||
@ -435,60 +513,24 @@ $loopLabel sty $indexVar
|
|||||||
}
|
}
|
||||||
asmgen.out(endLabel)
|
asmgen.out(endLabel)
|
||||||
}
|
}
|
||||||
DataType.ARRAY_UW_SPLIT, DataType.ARRAY_W_SPLIT -> {
|
iterableDt.isFloatArray -> {
|
||||||
val indexVar = asmgen.makeLabel("for_index")
|
|
||||||
val loopvarName = asmgen.asmVariableName(stmt.variable)
|
|
||||||
asmgen.out("""
|
|
||||||
ldy #0
|
|
||||||
$loopLabel sty $indexVar
|
|
||||||
lda ${iterableName}_lsb,y
|
|
||||||
sta $loopvarName
|
|
||||||
lda ${iterableName}_msb,y
|
|
||||||
sta $loopvarName+1""")
|
|
||||||
asmgen.translate(stmt.statements)
|
|
||||||
if(numElements<=255) {
|
|
||||||
asmgen.out("""
|
|
||||||
ldy $indexVar
|
|
||||||
iny
|
|
||||||
cpy #$numElements
|
|
||||||
beq $endLabel
|
|
||||||
bne $loopLabel""")
|
|
||||||
} else {
|
|
||||||
// length is 256
|
|
||||||
asmgen.out("""
|
|
||||||
ldy $indexVar
|
|
||||||
iny
|
|
||||||
bne $loopLabel
|
|
||||||
beq $endLabel""")
|
|
||||||
}
|
|
||||||
if(numElements>=16) {
|
|
||||||
// allocate index var on ZP if possible
|
|
||||||
val result = zeropage.allocate(indexVar, DataType.UBYTE, null, stmt.position, asmgen.errors)
|
|
||||||
result.fold(
|
|
||||||
success = { (address,_,_)-> asmgen.out("""$indexVar = $address ; auto zp UBYTE""") },
|
|
||||||
failure = { asmgen.out("$indexVar .byte 0") }
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
asmgen.out("$indexVar .byte 0")
|
|
||||||
}
|
|
||||||
asmgen.out(endLabel)
|
|
||||||
}
|
|
||||||
DataType.ARRAY_F -> {
|
|
||||||
throw AssemblyError("for loop with floating point variables is not supported")
|
throw AssemblyError("for loop with floating point variables is not supported")
|
||||||
}
|
}
|
||||||
else -> throw AssemblyError("can't iterate over $iterableDt")
|
else -> throw AssemblyError("can't iterate over $iterableDt")
|
||||||
}
|
}
|
||||||
asmgen.loopEndLabels.pop()
|
asmgen.loopEndLabels.removeLast()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun translateForOverConstRange(stmt: PtForLoop, iterableDt: DataType, range: IntProgression) {
|
private fun translateForOverConstRange(stmt: PtForLoop, iterableDt: DataType, range: IntProgression) {
|
||||||
if (range.isEmpty() || range.step==0)
|
if (range.isEmpty() || range.step==0)
|
||||||
throw AssemblyError("empty range or step 0")
|
throw AssemblyError("empty range or step 0")
|
||||||
if(iterableDt==DataType.ARRAY_B || iterableDt==DataType.ARRAY_UB) {
|
if(iterableDt.isByteArray) {
|
||||||
|
if(range.last==range.first) return translateForSimpleByteRangeAsc(stmt, range)
|
||||||
if(range.step==1 && range.last>range.first) return translateForSimpleByteRangeAsc(stmt, range)
|
if(range.step==1 && range.last>range.first) return translateForSimpleByteRangeAsc(stmt, range)
|
||||||
if(range.step==-1 && range.last<range.first) return translateForSimpleByteRangeDesc(stmt, range)
|
if(range.step==-1 && range.last<range.first) return translateForSimpleByteRangeDesc(stmt, range, iterableDt.isUnsignedByteArray)
|
||||||
}
|
}
|
||||||
else if(iterableDt==DataType.ARRAY_W || iterableDt==DataType.ARRAY_UW) {
|
else if(iterableDt.isWordArray) {
|
||||||
|
if(range.last==range.first) return translateForSimpleWordRangeAsc(stmt, range)
|
||||||
if(range.step==1 && range.last>range.first) return translateForSimpleWordRangeAsc(stmt, range)
|
if(range.step==1 && range.last>range.first) return translateForSimpleWordRangeAsc(stmt, range)
|
||||||
if(range.step==-1 && range.last<range.first) return translateForSimpleWordRangeDesc(stmt, range)
|
if(range.step==-1 && range.last<range.first) return translateForSimpleWordRangeDesc(stmt, range)
|
||||||
}
|
}
|
||||||
@ -496,9 +538,9 @@ $loopLabel sty $indexVar
|
|||||||
// not one of the easy cases, generate more complex code...
|
// not one of the easy cases, generate more complex code...
|
||||||
val loopLabel = asmgen.makeLabel("for_loop")
|
val loopLabel = asmgen.makeLabel("for_loop")
|
||||||
val endLabel = asmgen.makeLabel("for_end")
|
val endLabel = asmgen.makeLabel("for_end")
|
||||||
asmgen.loopEndLabels.push(endLabel)
|
asmgen.loopEndLabels.add(endLabel)
|
||||||
when(iterableDt) {
|
when {
|
||||||
DataType.ARRAY_B, DataType.ARRAY_UB -> {
|
iterableDt.isByteArray -> {
|
||||||
// loop over byte range via loopvar, step >= 2 or <= -2
|
// loop over byte range via loopvar, step >= 2 or <= -2
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
val varname = asmgen.asmVariableName(stmt.variable)
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
@ -508,7 +550,7 @@ $loopLabel""")
|
|||||||
asmgen.translate(stmt.statements)
|
asmgen.translate(stmt.statements)
|
||||||
when (range.step) {
|
when (range.step) {
|
||||||
0, 1, -1 -> {
|
0, 1, -1 -> {
|
||||||
throw AssemblyError("step 0, 1 and -1 should have been handled specifically $stmt")
|
throw AssemblyError("step 0, 1 and -1 should have been handled specifically $range ${stmt.position}")
|
||||||
}
|
}
|
||||||
2 -> {
|
2 -> {
|
||||||
if(range.last==255 || range.last==254) {
|
if(range.last==255 || range.last==254) {
|
||||||
@ -563,7 +605,7 @@ $loopLabel""")
|
|||||||
}
|
}
|
||||||
asmgen.out(endLabel)
|
asmgen.out(endLabel)
|
||||||
}
|
}
|
||||||
DataType.ARRAY_W, DataType.ARRAY_UW -> {
|
iterableDt.isWordArray && !iterableDt.isSplitWordArray -> {
|
||||||
// loop over word range via loopvar, step >= 2 or <= -2
|
// loop over word range via loopvar, step >= 2 or <= -2
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
val varname = asmgen.asmVariableName(stmt.variable)
|
||||||
when (range.step) {
|
when (range.step) {
|
||||||
@ -602,13 +644,13 @@ $loopLabel""")
|
|||||||
}
|
}
|
||||||
else -> throw AssemblyError("range expression can only be byte or word")
|
else -> throw AssemblyError("range expression can only be byte or word")
|
||||||
}
|
}
|
||||||
asmgen.loopEndLabels.pop()
|
asmgen.loopEndLabels.removeLast()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun translateForSimpleByteRangeAsc(stmt: PtForLoop, range: IntProgression) {
|
private fun translateForSimpleByteRangeAsc(stmt: PtForLoop, range: IntProgression) {
|
||||||
val loopLabel = asmgen.makeLabel("for_loop")
|
val loopLabel = asmgen.makeLabel("for_loop")
|
||||||
val endLabel = asmgen.makeLabel("for_end")
|
val endLabel = asmgen.makeLabel("for_end")
|
||||||
asmgen.loopEndLabels.push(endLabel)
|
asmgen.loopEndLabels.add(endLabel)
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
val varname = asmgen.asmVariableName(stmt.variable)
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda #${range.first}
|
lda #${range.first}
|
||||||
@ -628,50 +670,50 @@ $endLabel""")
|
|||||||
bne $loopLabel
|
bne $loopLabel
|
||||||
$endLabel""")
|
$endLabel""")
|
||||||
}
|
}
|
||||||
asmgen.loopEndLabels.pop()
|
asmgen.loopEndLabels.removeLast()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun translateForSimpleByteRangeDesc(stmt: PtForLoop, range: IntProgression) {
|
private fun translateForSimpleByteRangeDesc(stmt: PtForLoop, range: IntProgression, unsigned: Boolean) {
|
||||||
val loopLabel = asmgen.makeLabel("for_loop")
|
val loopLabel = asmgen.makeLabel("for_loop")
|
||||||
val endLabel = asmgen.makeLabel("for_end")
|
|
||||||
asmgen.loopEndLabels.push(endLabel)
|
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
val varname = asmgen.asmVariableName(stmt.variable)
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda #${range.first}
|
lda #${range.first}
|
||||||
sta $varname
|
sta $varname
|
||||||
$loopLabel""")
|
$loopLabel""")
|
||||||
asmgen.translate(stmt.statements)
|
asmgen.translate(stmt.statements)
|
||||||
when (range.last) {
|
when (range.last) {
|
||||||
0 -> {
|
0 -> {
|
||||||
asmgen.out("""
|
if(!unsigned || range.first<=127) {
|
||||||
lda $varname
|
asmgen.out("""
|
||||||
beq $endLabel
|
dec $varname
|
||||||
dec $varname""")
|
bpl $loopLabel""")
|
||||||
asmgen.jmp(loopLabel)
|
} else {
|
||||||
asmgen.out(endLabel)
|
asmgen.out("""
|
||||||
|
dec $varname
|
||||||
|
lda $varname
|
||||||
|
cmp #255
|
||||||
|
bne $loopLabel""")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
1 -> {
|
1 -> {
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
dec $varname
|
dec $varname
|
||||||
bne $loopLabel
|
bne $loopLabel""")
|
||||||
$endLabel""")
|
|
||||||
}
|
}
|
||||||
else -> {
|
else -> {
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
dec $varname
|
dec $varname
|
||||||
lda $varname
|
lda $varname
|
||||||
cmp #${range.last-1}
|
cmp #${range.last-1}
|
||||||
bne $loopLabel
|
bne $loopLabel""")
|
||||||
$endLabel""")
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
asmgen.loopEndLabels.pop()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun translateForSimpleWordRangeAsc(stmt: PtForLoop, range: IntProgression) {
|
private fun translateForSimpleWordRangeAsc(stmt: PtForLoop, range: IntProgression) {
|
||||||
val loopLabel = asmgen.makeLabel("for_loop")
|
val loopLabel = asmgen.makeLabel("for_loop")
|
||||||
val endLabel = asmgen.makeLabel("for_end")
|
val endLabel = asmgen.makeLabel("for_end")
|
||||||
asmgen.loopEndLabels.push(endLabel)
|
asmgen.loopEndLabels.add(endLabel)
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
val varname = asmgen.asmVariableName(stmt.variable)
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda #<${range.first}
|
lda #<${range.first}
|
||||||
@ -692,13 +734,13 @@ $loopLabel""")
|
|||||||
inc $varname+1""")
|
inc $varname+1""")
|
||||||
asmgen.jmp(loopLabel)
|
asmgen.jmp(loopLabel)
|
||||||
asmgen.out(endLabel)
|
asmgen.out(endLabel)
|
||||||
asmgen.loopEndLabels.pop()
|
asmgen.loopEndLabels.removeLast()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun translateForSimpleWordRangeDesc(stmt: PtForLoop, range: IntProgression) {
|
private fun translateForSimpleWordRangeDesc(stmt: PtForLoop, range: IntProgression) {
|
||||||
val loopLabel = asmgen.makeLabel("for_loop")
|
val loopLabel = asmgen.makeLabel("for_loop")
|
||||||
val endLabel = asmgen.makeLabel("for_end")
|
val endLabel = asmgen.makeLabel("for_end")
|
||||||
asmgen.loopEndLabels.push(endLabel)
|
asmgen.loopEndLabels.add(endLabel)
|
||||||
val varname = asmgen.asmVariableName(stmt.variable)
|
val varname = asmgen.asmVariableName(stmt.variable)
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda #<${range.first}
|
lda #<${range.first}
|
||||||
@ -707,20 +749,29 @@ $loopLabel""")
|
|||||||
sty $varname+1
|
sty $varname+1
|
||||||
$loopLabel""")
|
$loopLabel""")
|
||||||
asmgen.translate(stmt.statements)
|
asmgen.translate(stmt.statements)
|
||||||
|
if(range.last==0) {
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varname
|
||||||
|
bne ++
|
||||||
|
lda $varname+1
|
||||||
|
beq $endLabel""")
|
||||||
|
} else {
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varname
|
||||||
|
cmp #<${range.last}
|
||||||
|
bne +
|
||||||
|
lda $varname+1
|
||||||
|
cmp #>${range.last}
|
||||||
|
beq $endLabel""")
|
||||||
|
}
|
||||||
asmgen.out("""
|
asmgen.out("""
|
||||||
lda $varname
|
|
||||||
cmp #<${range.last}
|
|
||||||
bne +
|
|
||||||
lda $varname+1
|
|
||||||
cmp #>${range.last}
|
|
||||||
beq $endLabel
|
|
||||||
+ lda $varname
|
+ lda $varname
|
||||||
bne +
|
bne +
|
||||||
dec $varname+1
|
dec $varname+1
|
||||||
+ dec $varname""")
|
+ dec $varname""")
|
||||||
asmgen.jmp(loopLabel)
|
asmgen.jmp(loopLabel)
|
||||||
asmgen.out(endLabel)
|
asmgen.out(endLabel)
|
||||||
asmgen.loopEndLabels.pop()
|
asmgen.loopEndLabels.removeLast()
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun assignLoopvarWord(stmt: PtForLoop, range: PtRange) =
|
private fun assignLoopvarWord(stmt: PtForLoop, range: PtRange) =
|
||||||
|
@ -15,9 +15,12 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
// just ignore any result values from the function call.
|
// just ignore any result values from the function call.
|
||||||
}
|
}
|
||||||
|
|
||||||
internal fun optimizeIntArgsViaRegisters(sub: PtSub) =
|
internal fun optimizeIntArgsViaCpuRegisters(params: List<PtSubroutineParameter>) =
|
||||||
(sub.parameters.size==1 && sub.parameters[0].type in IntegerDatatypesWithBoolean)
|
when(params.size) {
|
||||||
|| (sub.parameters.size==2 && sub.parameters[0].type in ByteDatatypesWithBoolean && sub.parameters[1].type in ByteDatatypesWithBoolean)
|
1 -> params[0].type.isIntegerOrBool
|
||||||
|
2 -> params[0].type.isByteOrBool && params[1].type.isByteOrBool
|
||||||
|
else -> false
|
||||||
|
}
|
||||||
|
|
||||||
internal fun translateFunctionCall(call: PtFunctionCall) {
|
internal fun translateFunctionCall(call: PtFunctionCall) {
|
||||||
// Output only the code to set up the parameters and perform the actual call
|
// Output only the code to set up the parameters and perform the actual call
|
||||||
@ -40,27 +43,104 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
sub.children.forEach { asmgen.translate(it as PtInlineAssembly) }
|
sub.children.forEach { asmgen.translate(it as PtInlineAssembly) }
|
||||||
asmgen.out(" \t; inlined routine end: ${sub.name}")
|
asmgen.out(" \t; inlined routine end: ${sub.name}")
|
||||||
} else {
|
} else {
|
||||||
asmgen.out(" jsr $subAsmName")
|
val bank = sub.address?.constbank?.toString()
|
||||||
|
if(bank==null) {
|
||||||
|
val varbank = if(sub.address?.varbank==null) null else asmgen.asmVariableName(sub.address!!.varbank!!)
|
||||||
|
if(varbank!=null) {
|
||||||
|
when(asmgen.options.compTarget.name) {
|
||||||
|
"cx16" -> {
|
||||||
|
// JSRFAR can jump to a banked RAM address as well!
|
||||||
|
asmgen.out("""
|
||||||
|
php
|
||||||
|
pha
|
||||||
|
lda $varbank
|
||||||
|
sta +
|
||||||
|
pla
|
||||||
|
plp
|
||||||
|
jsr cx16.JSRFAR
|
||||||
|
.word $subAsmName ; ${sub.address!!.address.toHex()}
|
||||||
|
+ .byte 0 ; modified"""
|
||||||
|
)
|
||||||
|
}
|
||||||
|
"c64" -> {
|
||||||
|
asmgen.out("""
|
||||||
|
php
|
||||||
|
pha
|
||||||
|
lda $varbank
|
||||||
|
sta +
|
||||||
|
pla
|
||||||
|
plp
|
||||||
|
jsr c64.x16jsrfar
|
||||||
|
.word $subAsmName ; ${sub.address!!.address.toHex()}
|
||||||
|
+ .byte 0 ; modified"""
|
||||||
|
)
|
||||||
|
}
|
||||||
|
"c128" -> {
|
||||||
|
asmgen.out("""
|
||||||
|
php
|
||||||
|
pha
|
||||||
|
lda $varbank
|
||||||
|
sta +
|
||||||
|
pla
|
||||||
|
plp
|
||||||
|
jsr c128.x16jsrfar
|
||||||
|
.word $subAsmName ; ${sub.address!!.address.toHex()}
|
||||||
|
+ .byte 0 ; modified"""
|
||||||
|
)
|
||||||
|
}
|
||||||
|
else -> throw AssemblyError("callfar is not supported on the selected compilation target")
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
asmgen.out(" jsr $subAsmName")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
when(asmgen.options.compTarget.name) {
|
||||||
|
"cx16" -> {
|
||||||
|
// JSRFAR can jump to a banked RAM address as well!
|
||||||
|
asmgen.out("""
|
||||||
|
jsr cx16.JSRFAR
|
||||||
|
.word $subAsmName ; ${sub.address!!.address.toHex()}
|
||||||
|
.byte $bank"""
|
||||||
|
)
|
||||||
|
}
|
||||||
|
"c64" -> {
|
||||||
|
asmgen.out("""
|
||||||
|
jsr c64.x16jsrfar
|
||||||
|
.word $subAsmName ; ${sub.address!!.address.toHex()}
|
||||||
|
.byte $bank"""
|
||||||
|
)
|
||||||
|
}
|
||||||
|
"c128" -> {
|
||||||
|
asmgen.out("""
|
||||||
|
jsr c128.x16jsrfar
|
||||||
|
.word $subAsmName ; ${sub.address!!.address.toHex()}
|
||||||
|
.byte $bank"""
|
||||||
|
)
|
||||||
|
}
|
||||||
|
else -> throw AssemblyError("callfar is not supported on the selected compilation target")
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if(sub is PtSub) {
|
else if(sub is PtSub) {
|
||||||
if(optimizeIntArgsViaRegisters(sub)) {
|
if(optimizeIntArgsViaCpuRegisters(sub.parameters)) {
|
||||||
if(sub.parameters.size==1) {
|
// Note that if the args fit into cpu registers, we don't concern ourselves here
|
||||||
val register = if (sub.parameters[0].type in ByteDatatypesWithBoolean) RegisterOrPair.A else RegisterOrPair.AY
|
// if they should be put into regular subroutine parameter variables, or the R0-R15 register variables.
|
||||||
argumentViaRegister(sub, IndexedValue(0, sub.parameters[0]), call.args[0], register)
|
// That is now up to the subroutine itself.
|
||||||
} else {
|
useCpuRegistersForArgs(call.args, sub)
|
||||||
// 2 byte params, second in Y, first in A
|
|
||||||
argumentViaRegister(sub, IndexedValue(0, sub.parameters[0]), call.args[0], RegisterOrPair.A)
|
|
||||||
if(asmgen.needAsaveForExpr(call.args[1]))
|
|
||||||
asmgen.out(" pha")
|
|
||||||
argumentViaRegister(sub, IndexedValue(1, sub.parameters[1]), call.args[1], RegisterOrPair.Y)
|
|
||||||
if(asmgen.needAsaveForExpr(call.args[1]))
|
|
||||||
asmgen.out(" pla")
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
// arguments via variables
|
// arguments via variables
|
||||||
for(arg in sub.parameters.withIndex().zip(call.args))
|
val (normalParams, registerParams) = sub.parameters.withIndex().partition { it.value.register == null }
|
||||||
argumentViaVariable(sub, arg.first.value, arg.second)
|
if (normalParams.isNotEmpty()) {
|
||||||
|
for (arg in normalParams.zip(call.args))
|
||||||
|
argumentViaVariable(sub, arg.first.value, arg.second)
|
||||||
|
}
|
||||||
|
if (registerParams.isNotEmpty()) {
|
||||||
|
// the R0-R15 'registers' are not really registers. They're just special variables.
|
||||||
|
for (arg in registerParams.zip(call.args))
|
||||||
|
argumentViaVariable(sub, arg.first.value, arg.second)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
asmgen.out(" jsr $subAsmName")
|
asmgen.out(" jsr $subAsmName")
|
||||||
}
|
}
|
||||||
@ -69,11 +149,35 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
// remember: dealing with the X register and/or dealing with return values is the responsibility of the caller
|
// remember: dealing with the X register and/or dealing with return values is the responsibility of the caller
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun useCpuRegistersForArgs(args: List<PtExpression>, sub: PtSub) {
|
||||||
|
val params = sub.parameters
|
||||||
|
when(params.size) {
|
||||||
|
1 -> {
|
||||||
|
val register = if (params[0].type.isByteOrBool) RegisterOrPair.A else RegisterOrPair.AY
|
||||||
|
argumentViaRegister(sub, IndexedValue(0, params[0]), args[0], register)
|
||||||
|
}
|
||||||
|
2 -> {
|
||||||
|
if(params[0].type.isByteOrBool && params[1].type.isByteOrBool) {
|
||||||
|
// 2 byte params, second in Y, first in A
|
||||||
|
argumentViaRegister(sub, IndexedValue(0, params[0]), args[0], RegisterOrPair.A)
|
||||||
|
if(asmgen.needAsaveForExpr(args[1]))
|
||||||
|
asmgen.out(" pha")
|
||||||
|
argumentViaRegister(sub, IndexedValue(1, params[1]), args[1], RegisterOrPair.Y)
|
||||||
|
if(asmgen.needAsaveForExpr(args[1]))
|
||||||
|
asmgen.out(" pla")
|
||||||
|
} else {
|
||||||
|
throw AssemblyError("cannot use registers for word+byte")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else -> throw AssemblyError("cannot use cpu registers for >2 arguments")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
private fun usesOtherRegistersWhileEvaluating(arg: PtExpression): Boolean {
|
private fun usesOtherRegistersWhileEvaluating(arg: PtExpression): Boolean {
|
||||||
return when(arg) {
|
return when(arg) {
|
||||||
is PtBuiltinFunctionCall -> {
|
is PtBuiltinFunctionCall -> {
|
||||||
if (arg.name == "lsb" || arg.name == "msb")
|
if (arg.name in arrayOf("lsb", "msb", "lsw", "msw"))
|
||||||
return usesOtherRegistersWhileEvaluating(arg.args[0])
|
return usesOtherRegistersWhileEvaluating(arg.args[0])
|
||||||
if (arg.name == "mkword")
|
if (arg.name == "mkword")
|
||||||
return usesOtherRegistersWhileEvaluating(arg.args[0]) || usesOtherRegistersWhileEvaluating(arg.args[1])
|
return usesOtherRegistersWhileEvaluating(arg.args[0]) || usesOtherRegistersWhileEvaluating(arg.args[1])
|
||||||
@ -81,8 +185,8 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
}
|
}
|
||||||
is PtAddressOf -> false
|
is PtAddressOf -> false
|
||||||
is PtIdentifier -> false
|
is PtIdentifier -> false
|
||||||
is PtMachineRegister -> false
|
is PtIrRegister -> false
|
||||||
is PtMemoryByte -> return usesOtherRegistersWhileEvaluating(arg.address)
|
is PtMemoryByte -> arg.address !is PtNumber && arg.address !is PtIdentifier
|
||||||
is PtNumber -> false
|
is PtNumber -> false
|
||||||
is PtBool -> false
|
is PtBool -> false
|
||||||
else -> true
|
else -> true
|
||||||
@ -90,7 +194,7 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun argumentsViaRegisters(sub: PtAsmSub, call: PtFunctionCall) {
|
private fun argumentsViaRegisters(sub: PtAsmSub, call: PtFunctionCall) {
|
||||||
val registersUsed = mutableListOf<RegisterOrStatusflag>();
|
val registersUsed = mutableListOf<RegisterOrStatusflag>()
|
||||||
|
|
||||||
fun usedA() = registersUsed.any {it.registerOrPair==RegisterOrPair.A || it.registerOrPair==RegisterOrPair.AX || it.registerOrPair==RegisterOrPair.AY}
|
fun usedA() = registersUsed.any {it.registerOrPair==RegisterOrPair.A || it.registerOrPair==RegisterOrPair.AX || it.registerOrPair==RegisterOrPair.AY}
|
||||||
fun usedX() = registersUsed.any {it.registerOrPair==RegisterOrPair.X || it.registerOrPair==RegisterOrPair.AX || it.registerOrPair==RegisterOrPair.XY}
|
fun usedX() = registersUsed.any {it.registerOrPair==RegisterOrPair.X || it.registerOrPair==RegisterOrPair.AX || it.registerOrPair==RegisterOrPair.XY}
|
||||||
@ -110,9 +214,9 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
throw AssemblyError("call argument evaluation problem: can't save cpu statusregister parameter ${call.position}")
|
throw AssemblyError("call argument evaluation problem: can't save cpu statusregister parameter ${call.position}")
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if(usedX()) asmgen.saveRegisterStack(CpuRegister.X, false)
|
if(usedX()) asmgen.saveRegisterStack(CpuRegister.X, usedA())
|
||||||
if(usedY()) asmgen.saveRegisterStack(CpuRegister.Y, false)
|
if(usedY()) asmgen.saveRegisterStack(CpuRegister.Y, usedA())
|
||||||
if(usedA()) asmgen.saveRegisterStack(CpuRegister.A, false)
|
if(usedA()) asmgen.saveRegisterStack(CpuRegister.A, usedA())
|
||||||
val used = argumentViaRegister(sub, IndexedValue(it, param.second), arg)
|
val used = argumentViaRegister(sub, IndexedValue(it, param.second), arg)
|
||||||
if(usedA()) asmgen.restoreRegisterStack(CpuRegister.A, false)
|
if(usedA()) asmgen.restoreRegisterStack(CpuRegister.A, false)
|
||||||
if(usedY()) asmgen.restoreRegisterStack(CpuRegister.Y, true)
|
if(usedY()) asmgen.restoreRegisterStack(CpuRegister.Y, true)
|
||||||
@ -131,8 +235,15 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
if(!isArgumentTypeCompatible(value.type, parameter.type))
|
if(!isArgumentTypeCompatible(value.type, parameter.type))
|
||||||
throw AssemblyError("argument type incompatible")
|
throw AssemblyError("argument type incompatible")
|
||||||
|
|
||||||
val varName = asmgen.asmVariableName(sub.scopedName + "." + parameter.name)
|
val reg = parameter.register
|
||||||
asmgen.assignExpressionToVariable(value, varName, parameter.type)
|
if(reg!=null) {
|
||||||
|
require(reg in Cx16VirtualRegisters) { "can only use R0-R15 'registers' here" }
|
||||||
|
val varName = "cx16.${reg.name.lowercase()}"
|
||||||
|
asmgen.assignExpressionToVariable(value, varName, parameter.type)
|
||||||
|
} else {
|
||||||
|
val varName = asmgen.asmVariableName(sub.scopedName + "." + parameter.name)
|
||||||
|
asmgen.assignExpressionToVariable(value, varName, parameter.type)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun argumentViaRegister(sub: IPtSubroutine, parameter: IndexedValue<PtSubroutineParameter>, value: PtExpression, registerOverride: RegisterOrPair? = null): RegisterOrStatusflag {
|
private fun argumentViaRegister(sub: IPtSubroutine, parameter: IndexedValue<PtSubroutineParameter>, value: PtExpression, registerOverride: RegisterOrPair? = null): RegisterOrStatusflag {
|
||||||
@ -148,7 +259,7 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
val register = paramRegister.registerOrPair
|
val register = paramRegister.registerOrPair
|
||||||
val requiredDt = parameter.value.type
|
val requiredDt = parameter.value.type
|
||||||
if(requiredDt!=value.type) {
|
if(requiredDt!=value.type) {
|
||||||
if(value.type largerThan requiredDt)
|
if(value.type.largerSizeThan(requiredDt))
|
||||||
throw AssemblyError("can only convert byte values to word param types")
|
throw AssemblyError("can only convert byte values to word param types")
|
||||||
}
|
}
|
||||||
if (statusflag!=null) {
|
if (statusflag!=null) {
|
||||||
@ -187,25 +298,24 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
else {
|
else {
|
||||||
// via register or register pair
|
// via register or register pair
|
||||||
register!!
|
register!!
|
||||||
if(requiredDt largerThan value.type) {
|
if(requiredDt.largerSizeThan(value.type)) {
|
||||||
// we need to sign extend the source, do this via temporary word variable
|
// we need to sign extend the source, do this via temporary word variable
|
||||||
asmgen.assignExpressionToVariable(value, "P8ZP_SCRATCH_W1", DataType.UBYTE)
|
asmgen.assignExpressionToVariable(value, "P8ZP_SCRATCH_W1", DataType.forDt(BaseDataType.UBYTE))
|
||||||
asmgen.signExtendVariableLsb("P8ZP_SCRATCH_W1", value.type)
|
asmgen.signExtendVariableLsb("P8ZP_SCRATCH_W1", value.type.base)
|
||||||
asmgen.assignVariableToRegister("P8ZP_SCRATCH_W1", register, null, Position.DUMMY)
|
asmgen.assignVariableToRegister("P8ZP_SCRATCH_W1", register, null, Position.DUMMY)
|
||||||
} else {
|
} else {
|
||||||
val scope = value.definingISub()
|
val scope = value.definingISub()
|
||||||
val target: AsmAssignTarget =
|
val target: AsmAssignTarget =
|
||||||
if(parameter.value.type in ByteDatatypes && (register==RegisterOrPair.AX || register == RegisterOrPair.AY || register==RegisterOrPair.XY || register in Cx16VirtualRegisters))
|
if(parameter.value.type.isByte && (register==RegisterOrPair.AX || register == RegisterOrPair.AY || register==RegisterOrPair.XY || register in Cx16VirtualRegisters))
|
||||||
AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, parameter.value.type, scope, value.position, register = register)
|
AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, parameter.value.type, scope, value.position, register = register)
|
||||||
else {
|
else {
|
||||||
val signed = parameter.value.type == DataType.BYTE || parameter.value.type == DataType.WORD
|
AsmAssignTarget.fromRegisters(register, parameter.value.type.isSigned, value.position, scope, asmgen)
|
||||||
AsmAssignTarget.fromRegisters(register, signed, value.position, scope, asmgen)
|
|
||||||
}
|
}
|
||||||
val src = if(value.type in PassByReferenceDatatypes) {
|
val src = if(value.type.isPassByRef) {
|
||||||
if(value is PtIdentifier) {
|
if(value is PtIdentifier) {
|
||||||
val addr = PtAddressOf(Position.DUMMY)
|
val addr = PtAddressOf(Position.DUMMY)
|
||||||
addr.add(value)
|
addr.add(value)
|
||||||
addr.parent = sub as PtNode
|
addr.parent = scope as PtNode
|
||||||
AsmAssignSource.fromAstSource(addr, program, asmgen).adjustSignedUnsigned(target)
|
AsmAssignSource.fromAstSource(addr, program, asmgen).adjustSignedUnsigned(target)
|
||||||
} else {
|
} else {
|
||||||
AsmAssignSource.fromAstSource(value, program, asmgen).adjustSignedUnsigned(target)
|
AsmAssignSource.fromAstSource(value, program, asmgen).adjustSignedUnsigned(target)
|
||||||
@ -213,7 +323,7 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
} else {
|
} else {
|
||||||
AsmAssignSource.fromAstSource(value, program, asmgen).adjustSignedUnsigned(target)
|
AsmAssignSource.fromAstSource(value, program, asmgen).adjustSignedUnsigned(target)
|
||||||
}
|
}
|
||||||
asmgen.translateNormalAssignment(AsmAssignment(src, target, program.memsizer, Position.DUMMY), scope)
|
asmgen.translateNormalAssignment(AsmAssignment(src, target, program.memsizer, target.position), scope)
|
||||||
}
|
}
|
||||||
return RegisterOrStatusflag(register, null)
|
return RegisterOrStatusflag(register, null)
|
||||||
}
|
}
|
||||||
@ -222,18 +332,18 @@ internal class FunctionCallAsmGen(private val program: PtProgram, private val as
|
|||||||
private fun isArgumentTypeCompatible(argType: DataType, paramType: DataType): Boolean {
|
private fun isArgumentTypeCompatible(argType: DataType, paramType: DataType): Boolean {
|
||||||
if(argType isAssignableTo paramType)
|
if(argType isAssignableTo paramType)
|
||||||
return true
|
return true
|
||||||
if(argType==DataType.BOOL && paramType==DataType.BOOL)
|
if(argType.isBool && paramType.isBool)
|
||||||
return true
|
return true
|
||||||
if(argType in ByteDatatypes && paramType in ByteDatatypes)
|
if(argType.isByte && paramType.isByte)
|
||||||
return true
|
return true
|
||||||
if(argType in WordDatatypes && paramType in WordDatatypes)
|
if(argType.isWord && paramType.isWord)
|
||||||
return true
|
return true
|
||||||
|
|
||||||
// we have a special rule for some types.
|
// we have a special rule for some types.
|
||||||
// strings are assignable to UWORD, for example, and vice versa
|
// strings are assignable to UWORD, for example, and vice versa
|
||||||
if(argType==DataType.STR && paramType==DataType.UWORD)
|
if(argType.isString && paramType.isUnsignedWord)
|
||||||
return true
|
return true
|
||||||
if(argType==DataType.UWORD && paramType == DataType.STR)
|
if(argType.isUnsignedWord && paramType.isString)
|
||||||
return true
|
return true
|
||||||
|
|
||||||
return false
|
return false
|
||||||
|
File diff suppressed because it is too large
Load Diff
370
codeGenCpu6502/src/prog8/codegen/cpu6502/IfExpressionAsmGen.kt
Normal file
370
codeGenCpu6502/src/prog8/codegen/cpu6502/IfExpressionAsmGen.kt
Normal file
@ -0,0 +1,370 @@
|
|||||||
|
package prog8.codegen.cpu6502
|
||||||
|
|
||||||
|
import prog8.code.ast.PtBinaryExpression
|
||||||
|
import prog8.code.ast.PtBuiltinFunctionCall
|
||||||
|
import prog8.code.ast.PtExpression
|
||||||
|
import prog8.code.ast.PtIdentifier
|
||||||
|
import prog8.code.ast.PtIfExpression
|
||||||
|
import prog8.code.ast.PtNumber
|
||||||
|
import prog8.code.ast.PtPrefix
|
||||||
|
import prog8.code.core.AssemblyError
|
||||||
|
import prog8.code.core.BaseDataType
|
||||||
|
import prog8.code.core.CpuRegister
|
||||||
|
import prog8.code.core.DataType
|
||||||
|
import prog8.code.core.IErrorReporter
|
||||||
|
import prog8.code.core.LogicalOperators
|
||||||
|
import prog8.code.core.RegisterOrPair
|
||||||
|
import prog8.codegen.cpu6502.assignment.AsmAssignTarget
|
||||||
|
import prog8.codegen.cpu6502.assignment.AssignmentAsmGen
|
||||||
|
import prog8.codegen.cpu6502.assignment.TargetStorageKind
|
||||||
|
|
||||||
|
internal class IfExpressionAsmGen(private val asmgen: AsmGen6502Internal, private val assignmentAsmGen: AssignmentAsmGen, private val errors: IErrorReporter) {
|
||||||
|
|
||||||
|
internal fun assignIfExpression(target: AsmAssignTarget, expr: PtIfExpression) {
|
||||||
|
require(target.datatype==expr.type)
|
||||||
|
val falseLabel = asmgen.makeLabel("ifexpr_false")
|
||||||
|
val endLabel = asmgen.makeLabel("ifexpr_end")
|
||||||
|
evalIfExpressionConditonAndBranchWhenFalse(expr.condition, falseLabel)
|
||||||
|
when {
|
||||||
|
expr.type.isByteOrBool -> {
|
||||||
|
asmgen.assignExpressionToRegister(expr.truevalue, RegisterOrPair.A, false)
|
||||||
|
asmgen.jmp(endLabel)
|
||||||
|
asmgen.out(falseLabel)
|
||||||
|
asmgen.assignExpressionToRegister(expr.falsevalue, RegisterOrPair.A, false)
|
||||||
|
asmgen.out(endLabel)
|
||||||
|
assignmentAsmGen.assignRegisterByte(target, CpuRegister.A, false, false)
|
||||||
|
}
|
||||||
|
expr.type.isWord -> {
|
||||||
|
asmgen.assignExpressionToRegister(expr.truevalue, RegisterOrPair.AY, false)
|
||||||
|
asmgen.jmp(endLabel)
|
||||||
|
asmgen.out(falseLabel)
|
||||||
|
asmgen.assignExpressionToRegister(expr.falsevalue, RegisterOrPair.AY, false)
|
||||||
|
asmgen.out(endLabel)
|
||||||
|
assignmentAsmGen.assignRegisterpairWord(target, RegisterOrPair.AY)
|
||||||
|
}
|
||||||
|
expr.type.isFloat -> {
|
||||||
|
asmgen.assignExpressionToRegister(expr.truevalue, RegisterOrPair.FAC1, true)
|
||||||
|
asmgen.jmp(endLabel)
|
||||||
|
asmgen.out(falseLabel)
|
||||||
|
asmgen.assignExpressionToRegister(expr.falsevalue, RegisterOrPair.FAC1, true)
|
||||||
|
asmgen.out(endLabel)
|
||||||
|
asmgen.assignRegister(RegisterOrPair.FAC1, target)
|
||||||
|
}
|
||||||
|
else -> throw AssemblyError("weird dt")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun evalIfExpressionConditonAndBranchWhenFalse(condition: PtExpression, falseLabel: String) {
|
||||||
|
if (condition is PtBinaryExpression) {
|
||||||
|
val rightDt = condition.right.type
|
||||||
|
return when {
|
||||||
|
rightDt.isByteOrBool -> translateIfExpressionByteConditionBranch(condition, falseLabel)
|
||||||
|
rightDt.isWord -> translateIfExpressionWordConditionBranch(condition, falseLabel)
|
||||||
|
rightDt.isFloat -> translateIfExpressionFloatConditionBranch(condition, falseLabel)
|
||||||
|
else -> throw AssemblyError("weird dt")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if(condition is PtPrefix && condition.operator=="not") {
|
||||||
|
throw AssemblyError("not prefix in ifexpression should have been replaced by swapped values")
|
||||||
|
} else {
|
||||||
|
// 'simple' condition, check if it is a byte bittest
|
||||||
|
val bittest = condition as? PtBuiltinFunctionCall
|
||||||
|
if(bittest!=null && bittest.name.startsWith("prog8_ifelse_bittest_")) {
|
||||||
|
val variable = bittest.args[0] as PtIdentifier
|
||||||
|
val bitnumber = (bittest.args[1] as PtNumber).number.toInt()
|
||||||
|
val testForBitSet = bittest.name.endsWith("_set")
|
||||||
|
when (bitnumber) {
|
||||||
|
7 -> {
|
||||||
|
// test via bit + N flag
|
||||||
|
asmgen.out(" bit ${variable.name}")
|
||||||
|
if(testForBitSet) asmgen.out(" bpl $falseLabel")
|
||||||
|
else asmgen.out(" bmi $falseLabel")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
6 -> {
|
||||||
|
// test via bit + V flag
|
||||||
|
asmgen.out(" bit ${variable.name}")
|
||||||
|
if(testForBitSet) asmgen.out(" bvc $falseLabel")
|
||||||
|
else asmgen.out(" bvs $falseLabel")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
else -> throw AssemblyError("prog8_ifelse_bittest can only work on bits 7 and 6")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// the condition is "simple" enough to just assign its 0/1 value to a register and branch on that
|
||||||
|
asmgen.assignConditionValueToRegisterAndTest(condition)
|
||||||
|
asmgen.out(" beq $falseLabel")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateIfExpressionByteConditionBranch(condition: PtBinaryExpression, falseLabel: String) {
|
||||||
|
val signed = condition.left.type.isSigned
|
||||||
|
val constValue = condition.right.asConstInteger()
|
||||||
|
if(constValue==0) {
|
||||||
|
return translateIfCompareWithZeroByteBranch(condition, signed, falseLabel)
|
||||||
|
}
|
||||||
|
|
||||||
|
when(condition.operator) {
|
||||||
|
"==" -> {
|
||||||
|
// if X==value
|
||||||
|
asmgen.assignExpressionToRegister(condition.left, RegisterOrPair.A, signed)
|
||||||
|
asmgen.cmpAwithByteValue(condition.right, false)
|
||||||
|
asmgen.out(" bne $falseLabel")
|
||||||
|
}
|
||||||
|
"!=" -> {
|
||||||
|
// if X!=value
|
||||||
|
asmgen.assignExpressionToRegister(condition.left, RegisterOrPair.A, signed)
|
||||||
|
asmgen.cmpAwithByteValue(condition.right, false)
|
||||||
|
asmgen.out(" beq $falseLabel")
|
||||||
|
}
|
||||||
|
in LogicalOperators -> {
|
||||||
|
val regAtarget = AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, DataType.forDt(BaseDataType.BOOL), condition.definingISub(), condition.position, register=RegisterOrPair.A)
|
||||||
|
if (assignmentAsmGen.optimizedLogicalExpr(condition, regAtarget)) {
|
||||||
|
asmgen.out(" beq $falseLabel")
|
||||||
|
} else {
|
||||||
|
errors.warn("SLOW FALLBACK FOR 'IFEXPR' CODEGEN - ask for support", condition.position) // should not occur ;-)
|
||||||
|
asmgen.assignConditionValueToRegisterAndTest(condition)
|
||||||
|
asmgen.out(" beq $falseLabel")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else -> {
|
||||||
|
// TODO don't store condition as expression result but just use the flags, like a normal PtIfElse translation does
|
||||||
|
// TODO: special cases for <, <=, >, >= above.
|
||||||
|
asmgen.assignConditionValueToRegisterAndTest(condition)
|
||||||
|
asmgen.out(" beq $falseLabel")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateIfExpressionWordConditionBranch(condition: PtBinaryExpression, falseLabel: String) {
|
||||||
|
// TODO can we reuse this whole thing from IfElse ?
|
||||||
|
val constValue = condition.right.asConstInteger()
|
||||||
|
if(constValue!=null) {
|
||||||
|
if (constValue == 0) {
|
||||||
|
when (condition.operator) {
|
||||||
|
"==" -> return translateWordExprIsZero(condition.left, falseLabel)
|
||||||
|
"!=" -> return translateWordExprIsNotZero(condition.left, falseLabel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (constValue != 0) {
|
||||||
|
when (condition.operator) {
|
||||||
|
"==" -> return translateWordExprEqualsNumber(condition.left, constValue, falseLabel)
|
||||||
|
"!=" -> return translateWordExprNotEqualsNumber(condition.left, constValue, falseLabel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
val variable = condition.right as? PtIdentifier
|
||||||
|
if(variable!=null) {
|
||||||
|
when (condition.operator) {
|
||||||
|
"==" -> return translateWordExprEqualsVariable(condition.left, variable, falseLabel)
|
||||||
|
"!=" -> return translateWordExprNotEqualsVariable(condition.left, variable, falseLabel)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO don't store condition as expression result but just use the flags, like a normal PtIfElse translation does
|
||||||
|
asmgen.assignConditionValueToRegisterAndTest(condition)
|
||||||
|
asmgen.out(" beq $falseLabel")
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateIfExpressionFloatConditionBranch(condition: PtBinaryExpression, elseLabel: String) {
|
||||||
|
val constValue = (condition.right as? PtNumber)?.number
|
||||||
|
if(constValue==0.0) {
|
||||||
|
if (condition.operator == "==") {
|
||||||
|
// if FL==0.0
|
||||||
|
asmgen.assignExpressionToRegister(condition.left, RegisterOrPair.FAC1, true)
|
||||||
|
asmgen.out(" jsr floats.SIGN | cmp #0 | bne $elseLabel")
|
||||||
|
return
|
||||||
|
} else if(condition.operator=="!=") {
|
||||||
|
// if FL!=0.0
|
||||||
|
asmgen.assignExpressionToRegister(condition.left, RegisterOrPair.FAC1, true)
|
||||||
|
asmgen.out(" jsr floats.SIGN | cmp #0 | beq $elseLabel")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
when(condition.operator) {
|
||||||
|
"==" -> {
|
||||||
|
asmgen.translateFloatsEqualsConditionIntoA(condition.left, condition.right)
|
||||||
|
asmgen.out(" beq $elseLabel")
|
||||||
|
}
|
||||||
|
"!=" -> {
|
||||||
|
asmgen.translateFloatsEqualsConditionIntoA(condition.left, condition.right)
|
||||||
|
asmgen.out(" bne $elseLabel")
|
||||||
|
}
|
||||||
|
"<" -> {
|
||||||
|
asmgen.translateFloatsLessConditionIntoA(condition.left, condition.right, false)
|
||||||
|
asmgen.out(" beq $elseLabel")
|
||||||
|
}
|
||||||
|
"<=" -> {
|
||||||
|
asmgen.translateFloatsLessConditionIntoA(condition.left, condition.right, true)
|
||||||
|
asmgen.out(" beq $elseLabel")
|
||||||
|
}
|
||||||
|
">" -> {
|
||||||
|
asmgen.translateFloatsLessConditionIntoA(condition.left, condition.right, true)
|
||||||
|
asmgen.out(" bne $elseLabel")
|
||||||
|
}
|
||||||
|
">=" -> {
|
||||||
|
asmgen.translateFloatsLessConditionIntoA(condition.left, condition.right, false)
|
||||||
|
asmgen.out(" bne $elseLabel")
|
||||||
|
}
|
||||||
|
else -> throw AssemblyError("expected comparison operator")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateWordExprNotEqualsVariable(expr: PtExpression, variable: PtIdentifier, falseLabel: String) {
|
||||||
|
// if w!=variable
|
||||||
|
// TODO reuse code from ifElse?
|
||||||
|
val varRight = asmgen.asmVariableName(variable)
|
||||||
|
if(expr is PtIdentifier) {
|
||||||
|
val varLeft = asmgen.asmVariableName(expr)
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varLeft
|
||||||
|
cmp $varRight
|
||||||
|
bne +
|
||||||
|
lda $varLeft+1
|
||||||
|
cmp $varRight+1
|
||||||
|
beq $falseLabel
|
||||||
|
+""")
|
||||||
|
} else {
|
||||||
|
asmgen.assignExpressionToRegister(expr, RegisterOrPair.AY, false)
|
||||||
|
asmgen.out("""
|
||||||
|
cmp $varRight
|
||||||
|
bne +
|
||||||
|
cpy $varRight+1
|
||||||
|
beq $falseLabel
|
||||||
|
+""")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateWordExprEqualsVariable(expr: PtExpression, variable: PtIdentifier, falseLabel: String) {
|
||||||
|
// if w==variable
|
||||||
|
// TODO reuse code from ifElse?
|
||||||
|
val varRight = asmgen.asmVariableName(variable)
|
||||||
|
if(expr is PtIdentifier) {
|
||||||
|
val varLeft = asmgen.asmVariableName(expr)
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varLeft
|
||||||
|
cmp $varRight
|
||||||
|
bne $falseLabel
|
||||||
|
lda $varLeft+1
|
||||||
|
cmp $varRight+1
|
||||||
|
bne $falseLabel""")
|
||||||
|
} else {
|
||||||
|
asmgen.assignExpressionToRegister(expr, RegisterOrPair.AY, false)
|
||||||
|
asmgen.out("""
|
||||||
|
cmp $varRight
|
||||||
|
bne $falseLabel
|
||||||
|
cpy $varRight+1
|
||||||
|
bne $falseLabel""")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateWordExprNotEqualsNumber(expr: PtExpression, number: Int, falseLabel: String) {
|
||||||
|
// if w!=number
|
||||||
|
// TODO reuse code from ifElse?
|
||||||
|
if(expr is PtIdentifier) {
|
||||||
|
val varname = asmgen.asmVariableName(expr)
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varname
|
||||||
|
cmp #<$number
|
||||||
|
bne +
|
||||||
|
lda $varname+1
|
||||||
|
cmp #>$number
|
||||||
|
beq $falseLabel
|
||||||
|
+""")
|
||||||
|
} else {
|
||||||
|
asmgen.assignExpressionToRegister(expr, RegisterOrPair.AY, false)
|
||||||
|
asmgen.out("""
|
||||||
|
cmp #<$number
|
||||||
|
bne +
|
||||||
|
cpy #>$number
|
||||||
|
beq $falseLabel
|
||||||
|
+""")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateWordExprEqualsNumber(expr: PtExpression, number: Int, falseLabel: String) {
|
||||||
|
// if w==number
|
||||||
|
// TODO reuse code from ifElse?
|
||||||
|
if(expr is PtIdentifier) {
|
||||||
|
val varname = asmgen.asmVariableName(expr)
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varname
|
||||||
|
cmp #<$number
|
||||||
|
bne $falseLabel
|
||||||
|
lda $varname+1
|
||||||
|
cmp #>$number
|
||||||
|
bne $falseLabel""")
|
||||||
|
} else {
|
||||||
|
asmgen.assignExpressionToRegister(expr, RegisterOrPair.AY, false)
|
||||||
|
asmgen.out( """
|
||||||
|
cmp #<$number
|
||||||
|
bne $falseLabel
|
||||||
|
cpy #>$number
|
||||||
|
bne $falseLabel""")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateWordExprIsNotZero(expr: PtExpression, falseLabel: String) {
|
||||||
|
// if w!=0
|
||||||
|
// TODO reuse code from ifElse?
|
||||||
|
if(expr is PtIdentifier) {
|
||||||
|
val varname = asmgen.asmVariableName(expr)
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varname
|
||||||
|
ora $varname+1
|
||||||
|
beq $falseLabel""")
|
||||||
|
} else {
|
||||||
|
asmgen.assignExpressionToRegister(expr, RegisterOrPair.AY, false)
|
||||||
|
asmgen.out(" sty P8ZP_SCRATCH_REG | ora P8ZP_SCRATCH_REG | beq $falseLabel")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateWordExprIsZero(expr: PtExpression, falseLabel: String) {
|
||||||
|
// if w==0
|
||||||
|
// TODO reuse code from ifElse?
|
||||||
|
if(expr is PtIdentifier) {
|
||||||
|
val varname = asmgen.asmVariableName(expr)
|
||||||
|
asmgen.out("""
|
||||||
|
lda $varname
|
||||||
|
ora $varname+1
|
||||||
|
bne $falseLabel""")
|
||||||
|
} else {
|
||||||
|
asmgen.assignExpressionToRegister(expr, RegisterOrPair.AY, false)
|
||||||
|
asmgen.out(" sty P8ZP_SCRATCH_REG | ora P8ZP_SCRATCH_REG | bne $falseLabel")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private fun translateIfCompareWithZeroByteBranch(condition: PtBinaryExpression, signed: Boolean, falseLabel: String) {
|
||||||
|
// optimized code for byte comparisons with 0
|
||||||
|
asmgen.assignConditionValueToRegisterAndTest(condition.left)
|
||||||
|
when (condition.operator) {
|
||||||
|
"==" -> asmgen.out(" bne $falseLabel")
|
||||||
|
"!=" -> asmgen.out(" beq $falseLabel")
|
||||||
|
">" -> {
|
||||||
|
if(signed) asmgen.out(" bmi $falseLabel | beq $falseLabel")
|
||||||
|
else asmgen.out(" beq $falseLabel")
|
||||||
|
}
|
||||||
|
">=" -> {
|
||||||
|
if(signed) asmgen.out(" bmi $falseLabel")
|
||||||
|
else { /* always true for unsigned */ }
|
||||||
|
}
|
||||||
|
"<" -> {
|
||||||
|
if(signed) asmgen.out(" bpl $falseLabel")
|
||||||
|
else asmgen.jmp(falseLabel)
|
||||||
|
}
|
||||||
|
"<=" -> {
|
||||||
|
if(signed) {
|
||||||
|
// inverted '>'
|
||||||
|
asmgen.out("""
|
||||||
|
beq +
|
||||||
|
bpl $falseLabel
|
||||||
|
+""")
|
||||||
|
} else asmgen.out(" bne $falseLabel")
|
||||||
|
}
|
||||||
|
else -> throw AssemblyError("expected comparison operator")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -32,10 +32,6 @@ internal class ProgramAndVarsGen(
|
|||||||
|
|
||||||
internal fun generate() {
|
internal fun generate() {
|
||||||
header()
|
header()
|
||||||
val allBlocks = program.allBlocks()
|
|
||||||
|
|
||||||
if(allBlocks.first().name != "p8b_main" && allBlocks.first().name != "main")
|
|
||||||
throw AssemblyError("first block should be 'main' or 'p8b_main'")
|
|
||||||
|
|
||||||
if(errors.noErrors()) {
|
if(errors.noErrors()) {
|
||||||
program.allBlocks().forEach { block2asm(it) }
|
program.allBlocks().forEach { block2asm(it) }
|
||||||
@ -75,6 +71,12 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out("P8ZP_SCRATCH_REG = ${zp.SCRATCH_REG}")
|
asmgen.out("P8ZP_SCRATCH_REG = ${zp.SCRATCH_REG}")
|
||||||
asmgen.out("P8ZP_SCRATCH_W1 = ${zp.SCRATCH_W1} ; word")
|
asmgen.out("P8ZP_SCRATCH_W1 = ${zp.SCRATCH_W1} ; word")
|
||||||
asmgen.out("P8ZP_SCRATCH_W2 = ${zp.SCRATCH_W2} ; word")
|
asmgen.out("P8ZP_SCRATCH_W2 = ${zp.SCRATCH_W2} ; word")
|
||||||
|
if(compTarget.name=="c64") {
|
||||||
|
if(options.floats)
|
||||||
|
asmgen.out("PROG8_C64_BANK_CONFIG=31 ; basic+IO+kernal")
|
||||||
|
else
|
||||||
|
asmgen.out("PROG8_C64_BANK_CONFIG=30 ; IO+kernal, no basic")
|
||||||
|
}
|
||||||
asmgen.out(".weak") // hack to allow user to override the following two with command line redefinition (however, just use '-esa' command line option instead!)
|
asmgen.out(".weak") // hack to allow user to override the following two with command line redefinition (however, just use '-esa' command line option instead!)
|
||||||
asmgen.out(".endweak")
|
asmgen.out(".endweak")
|
||||||
|
|
||||||
@ -89,6 +91,13 @@ internal class ProgramAndVarsGen(
|
|||||||
OutputType.RAW -> {
|
OutputType.RAW -> {
|
||||||
asmgen.out("; ---- raw assembler program ----")
|
asmgen.out("; ---- raw assembler program ----")
|
||||||
asmgen.out("* = ${options.loadAddress.toHex()}")
|
asmgen.out("* = ${options.loadAddress.toHex()}")
|
||||||
|
asmgen.out("prog8_program_start\t; start of program label")
|
||||||
|
asmgen.out(" cld")
|
||||||
|
asmgen.out(" tsx ; save stackpointer for sys.exit()")
|
||||||
|
asmgen.out(" stx prog8_lib.orig_stackpointer")
|
||||||
|
if(!options.noSysInit)
|
||||||
|
asmgen.out(" jsr p8_sys_startup.init_system")
|
||||||
|
asmgen.out(" jsr p8_sys_startup.init_system_phase2")
|
||||||
}
|
}
|
||||||
OutputType.PRG -> {
|
OutputType.PRG -> {
|
||||||
when(options.launcher) {
|
when(options.launcher) {
|
||||||
@ -98,30 +107,43 @@ internal class ProgramAndVarsGen(
|
|||||||
}
|
}
|
||||||
asmgen.out("; ---- basic program with sys call ----")
|
asmgen.out("; ---- basic program with sys call ----")
|
||||||
asmgen.out("* = ${options.loadAddress.toHex()}")
|
asmgen.out("* = ${options.loadAddress.toHex()}")
|
||||||
|
asmgen.out("prog8_program_start\t; start of program label")
|
||||||
val year = LocalDate.now().year
|
val year = LocalDate.now().year
|
||||||
asmgen.out(" .word (+), $year")
|
asmgen.out(" .word (+), $year")
|
||||||
asmgen.out(" .null $9e, format(' %d ', prog8_entrypoint), $3a, $8f, ' prog8'")
|
asmgen.out(" .null $9e, format(' %d ', prog8_entrypoint), $3a, $8f, ' prog8'")
|
||||||
asmgen.out("+\t.word 0")
|
asmgen.out("+\t.word 0")
|
||||||
asmgen.out("prog8_entrypoint\t; assembly code starts here")
|
asmgen.out("prog8_entrypoint")
|
||||||
|
asmgen.out(" cld")
|
||||||
|
asmgen.out(" tsx ; save stackpointer for sys.exit()")
|
||||||
|
asmgen.out(" stx prog8_lib.orig_stackpointer")
|
||||||
if(!options.noSysInit)
|
if(!options.noSysInit)
|
||||||
asmgen.out(" jsr sys.init_system")
|
asmgen.out(" jsr p8_sys_startup.init_system")
|
||||||
asmgen.out(" jsr sys.init_system_phase2")
|
asmgen.out(" jsr p8_sys_startup.init_system_phase2")
|
||||||
}
|
}
|
||||||
CbmPrgLauncherType.NONE -> {
|
CbmPrgLauncherType.NONE -> {
|
||||||
|
// this is the same as RAW
|
||||||
asmgen.out("; ---- program without basic sys call ----")
|
asmgen.out("; ---- program without basic sys call ----")
|
||||||
asmgen.out("* = ${options.loadAddress.toHex()}")
|
asmgen.out("* = ${options.loadAddress.toHex()}")
|
||||||
|
asmgen.out("prog8_program_start\t; start of program label")
|
||||||
|
asmgen.out(" cld")
|
||||||
|
asmgen.out(" tsx ; save stackpointer for sys.exit()")
|
||||||
|
asmgen.out(" stx prog8_lib.orig_stackpointer")
|
||||||
if(!options.noSysInit)
|
if(!options.noSysInit)
|
||||||
asmgen.out(" jsr sys.init_system")
|
asmgen.out(" jsr p8_sys_startup.init_system")
|
||||||
asmgen.out(" jsr sys.init_system_phase2")
|
asmgen.out(" jsr p8_sys_startup.init_system_phase2")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
OutputType.XEX -> {
|
OutputType.XEX -> {
|
||||||
asmgen.out("; ---- atari xex program ----")
|
asmgen.out("; ---- atari xex program ----")
|
||||||
asmgen.out("* = ${options.loadAddress.toHex()}")
|
asmgen.out("* = ${options.loadAddress.toHex()}")
|
||||||
|
asmgen.out("prog8_program_start\t; start of program label")
|
||||||
|
asmgen.out(" cld")
|
||||||
|
asmgen.out(" tsx ; save stackpointer for sys.exit()")
|
||||||
|
asmgen.out(" stx prog8_lib.orig_stackpointer")
|
||||||
if(!options.noSysInit)
|
if(!options.noSysInit)
|
||||||
asmgen.out(" jsr sys.init_system")
|
asmgen.out(" jsr p8_sys_startup.init_system")
|
||||||
asmgen.out(" jsr sys.init_system_phase2")
|
asmgen.out(" jsr p8_sys_startup.init_system_phase2")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -139,17 +161,20 @@ internal class ProgramAndVarsGen(
|
|||||||
if(options.floats)
|
if(options.floats)
|
||||||
asmgen.out(" lda #4 | sta $01") // to use floats, make sure Basic rom is banked in
|
asmgen.out(" lda #4 | sta $01") // to use floats, make sure Basic rom is banked in
|
||||||
asmgen.out(" jsr p8b_main.p8s_start")
|
asmgen.out(" jsr p8b_main.p8s_start")
|
||||||
asmgen.out(" jmp sys.cleanup_at_exit")
|
asmgen.out(" jmp p8_sys_startup.cleanup_at_exit")
|
||||||
}
|
}
|
||||||
"c64" -> {
|
"c64" -> {
|
||||||
asmgen.out(" jsr p8b_main.p8s_start | lda #31 | sta $01")
|
asmgen.out(" jsr p8b_main.p8s_start")
|
||||||
asmgen.out(" jmp sys.cleanup_at_exit")
|
asmgen.out(" jmp p8_sys_startup.cleanup_at_exit")
|
||||||
}
|
}
|
||||||
"c128" -> {
|
"c128" -> {
|
||||||
asmgen.out(" jsr p8b_main.p8s_start | lda #0 | sta ${"$"}ff00")
|
asmgen.out(" jsr p8b_main.p8s_start")
|
||||||
asmgen.out(" jmp sys.cleanup_at_exit")
|
asmgen.out(" jmp p8_sys_startup.cleanup_at_exit")
|
||||||
|
}
|
||||||
|
else -> {
|
||||||
|
asmgen.out(" jsr p8b_main.p8s_start")
|
||||||
|
asmgen.out(" jmp p8_sys_startup.cleanup_at_exit")
|
||||||
}
|
}
|
||||||
else -> asmgen.jmp("p8b_main.p8s_start")
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -173,12 +198,12 @@ internal class ProgramAndVarsGen(
|
|||||||
for(num in 1..count) {
|
for(num in 1..count) {
|
||||||
val name = asmgen.buildTempVarName(dt, num)
|
val name = asmgen.buildTempVarName(dt, num)
|
||||||
when (dt) {
|
when (dt) {
|
||||||
DataType.BOOL -> asmgen.out("$name .byte ?")
|
BaseDataType.BOOL -> asmgen.out("$name .byte ?")
|
||||||
DataType.BYTE -> asmgen.out("$name .char ?")
|
BaseDataType.BYTE -> asmgen.out("$name .char ?")
|
||||||
DataType.UBYTE -> asmgen.out("$name .byte ?")
|
BaseDataType.UBYTE -> asmgen.out("$name .byte ?")
|
||||||
DataType.WORD -> asmgen.out("$name .sint ?")
|
BaseDataType.WORD -> asmgen.out("$name .sint ?")
|
||||||
DataType.UWORD -> asmgen.out("$name .word ?")
|
BaseDataType.UWORD -> asmgen.out("$name .word ?")
|
||||||
DataType.FLOAT -> asmgen.out("$name .fill ${options.compTarget.machine.FLOAT_MEM_SIZE}")
|
BaseDataType.FLOAT -> asmgen.out("$name .fill ${options.compTarget.machine.FLOAT_MEM_SIZE}")
|
||||||
else -> throw AssemblyError("weird dt for extravar $dt")
|
else -> throw AssemblyError("weird dt for extravar $dt")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -265,6 +290,8 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out(" .cerror * > ${relocatedBssEnd.toHex()}, \"too many data for slabs_BSS section\"")
|
asmgen.out(" .cerror * > ${relocatedBssEnd.toHex()}, \"too many data for slabs_BSS section\"")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
asmgen.out(" ; memtop check")
|
||||||
|
asmgen.out(" .cerror * >= ${options.memtopAddress.toHex()}, \"Program too long by \", * - ${(options.memtopAddress-1u).toHex()}, \" bytes, memtop=${options.memtopAddress.toHex()}\"")
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun block2asm(block: PtBlock) {
|
private fun block2asm(block: PtBlock) {
|
||||||
@ -272,12 +299,6 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out("; ---- block: '${block.name}' ----")
|
asmgen.out("; ---- block: '${block.name}' ----")
|
||||||
if(block.options.address!=null)
|
if(block.options.address!=null)
|
||||||
asmgen.out("* = ${block.options.address!!.toHex()}")
|
asmgen.out("* = ${block.options.address!!.toHex()}")
|
||||||
else {
|
|
||||||
if(block.options.alignment==PtBlock.BlockAlignment.WORD)
|
|
||||||
asmgen.out("\t.align 2")
|
|
||||||
else if(block.options.alignment==PtBlock.BlockAlignment.PAGE)
|
|
||||||
asmgen.out("\t.align $100")
|
|
||||||
}
|
|
||||||
|
|
||||||
asmgen.out("${block.name}\t" + (if(block.options.forceOutput) ".block" else ".proc"))
|
asmgen.out("${block.name}\t" + (if(block.options.forceOutput) ".block" else ".proc"))
|
||||||
asmgen.outputSourceLine(block)
|
asmgen.outputSourceLine(block)
|
||||||
@ -400,22 +421,35 @@ internal class ProgramAndVarsGen(
|
|||||||
if((sub.name=="start" || sub.name=="p8s_start") && (sub.definingBlock()!!.name=="main" || sub.definingBlock()!!.name=="p8b_main"))
|
if((sub.name=="start" || sub.name=="p8s_start") && (sub.definingBlock()!!.name=="main" || sub.definingBlock()!!.name=="p8b_main"))
|
||||||
entrypointInitialization()
|
entrypointInitialization()
|
||||||
|
|
||||||
if(functioncallAsmGen.optimizeIntArgsViaRegisters(sub)) {
|
val params = sub.parameters
|
||||||
asmgen.out("; simple int arg(s) passed via register(s)")
|
if(functioncallAsmGen.optimizeIntArgsViaCpuRegisters(params)) {
|
||||||
if(sub.parameters.size==1) {
|
asmgen.out("; simple int arg(s) passed via cpu register(s)")
|
||||||
val dt = sub.parameters[0].type
|
|
||||||
val target = AsmAssignTarget(TargetStorageKind.VARIABLE, asmgen, dt, sub, sub.parameters[0].position, variableAsmName = sub.parameters[0].name)
|
fun varname(param: PtSubroutineParameter): String =
|
||||||
if(dt in ByteDatatypesWithBoolean)
|
if(param.register==null)
|
||||||
asmgen.assignRegister(RegisterOrPair.A, target)
|
param.name
|
||||||
else
|
else
|
||||||
asmgen.assignRegister(RegisterOrPair.AY, target)
|
param.register!!.asScopedNameVirtualReg(param.type).joinToString(".")
|
||||||
} else {
|
|
||||||
require(sub.parameters.size==2)
|
when(params.size) {
|
||||||
// 2 simple byte args, first in A, second in Y
|
1 -> {
|
||||||
val target1 = AsmAssignTarget(TargetStorageKind.VARIABLE, asmgen, sub.parameters[0].type, sub, sub.parameters[0].position, variableAsmName = sub.parameters[0].name)
|
val dt = params[0].type
|
||||||
val target2 = AsmAssignTarget(TargetStorageKind.VARIABLE, asmgen, sub.parameters[1].type, sub, sub.parameters[1].position, variableAsmName = sub.parameters[1].name)
|
val target = AsmAssignTarget(TargetStorageKind.VARIABLE, asmgen, dt, sub, params[0].position, variableAsmName = varname(params[0]))
|
||||||
asmgen.assignRegister(RegisterOrPair.A, target1)
|
if(dt.isByteOrBool)
|
||||||
asmgen.assignRegister(RegisterOrPair.Y, target2)
|
asmgen.assignRegister(RegisterOrPair.A, target) // single byte in A
|
||||||
|
else
|
||||||
|
asmgen.assignRegister(RegisterOrPair.AY, target) // word in AY
|
||||||
|
}
|
||||||
|
2 -> {
|
||||||
|
val target1 = AsmAssignTarget(TargetStorageKind.VARIABLE, asmgen, params[0].type, sub, params[0].position, variableAsmName = varname(params[0]))
|
||||||
|
val target2 = AsmAssignTarget(TargetStorageKind.VARIABLE, asmgen, params[1].type, sub, params[1].position, variableAsmName = varname(params[1]))
|
||||||
|
if(params[0].type.isByteOrBool && params[1].type.isByteOrBool) {
|
||||||
|
// 2 byte args, first in A, second in Y
|
||||||
|
asmgen.assignRegister(RegisterOrPair.A, target1)
|
||||||
|
asmgen.assignRegister(RegisterOrPair.Y, target2)
|
||||||
|
} else throw AssemblyError("cannot use registers for word+byte args")
|
||||||
|
}
|
||||||
|
else -> throw AssemblyError("cannot use registers for >2 args")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -429,9 +463,9 @@ internal class ProgramAndVarsGen(
|
|||||||
if(addr!=null)
|
if(addr!=null)
|
||||||
asmgen.out("$name = $addr")
|
asmgen.out("$name = $addr")
|
||||||
else when(dt) {
|
else when(dt) {
|
||||||
DataType.UBYTE -> asmgen.out("$name .byte ?")
|
BaseDataType.UBYTE -> asmgen.out("$name .byte ?")
|
||||||
DataType.UWORD -> asmgen.out("$name .word ?")
|
BaseDataType.UWORD -> asmgen.out("$name .word ?")
|
||||||
DataType.FLOAT -> asmgen.out("$name .fill ${options.compTarget.machine.FLOAT_MEM_SIZE}")
|
BaseDataType.FLOAT -> asmgen.out("$name .fill ${options.compTarget.machine.FLOAT_MEM_SIZE}")
|
||||||
else -> throw AssemblyError("weird dt for extravar $dt")
|
else -> throw AssemblyError("weird dt for extravar $dt")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -451,22 +485,11 @@ internal class ProgramAndVarsGen(
|
|||||||
}
|
}
|
||||||
|
|
||||||
private fun entrypointInitialization() {
|
private fun entrypointInitialization() {
|
||||||
asmgen.out("; program startup initialization")
|
// zero out the BSS area first, before setting the variable init values
|
||||||
asmgen.out(" cld | tsx | stx prog8_lib.orig_stackpointer ; required for sys.exit()")
|
// this is mainly to make sure the arrays are all zero'd out at program startup
|
||||||
// set full BSS area to zero
|
asmgen.out(" jsr prog8_lib.program_startup_clear_bss")
|
||||||
asmgen.out("""
|
|
||||||
.if prog8_bss_section_size>0
|
|
||||||
; reset all variables in BSS section to zero
|
|
||||||
lda #<prog8_bss_section_start
|
|
||||||
ldy #>prog8_bss_section_start
|
|
||||||
sta P8ZP_SCRATCH_W1
|
|
||||||
sty P8ZP_SCRATCH_W1+1
|
|
||||||
ldx #<prog8_bss_section_size
|
|
||||||
ldy #>prog8_bss_section_size
|
|
||||||
lda #0
|
|
||||||
jsr prog8_lib.memset
|
|
||||||
.endif""")
|
|
||||||
|
|
||||||
|
// initialize block-level (global) variables at program start
|
||||||
blockVariableInitializers.forEach {
|
blockVariableInitializers.forEach {
|
||||||
if (it.value.isNotEmpty())
|
if (it.value.isNotEmpty())
|
||||||
asmgen.out(" jsr ${it.key.name}.prog8_init_vars")
|
asmgen.out(" jsr ${it.key.name}.prog8_init_vars")
|
||||||
@ -509,17 +532,15 @@ internal class ProgramAndVarsGen(
|
|||||||
|
|
||||||
stringVarsWithInitInZp.forEach {
|
stringVarsWithInitInZp.forEach {
|
||||||
val varname = asmgen.asmVariableName(it.name)+"_init_value"
|
val varname = asmgen.asmVariableName(it.name)+"_init_value"
|
||||||
outputStringvar(varname, it.value.second, it.value.first)
|
outputStringvar(varname, 0, it.value.second, it.value.first)
|
||||||
}
|
}
|
||||||
|
|
||||||
arrayVarsWithInitInZp.forEach {
|
arrayVarsWithInitInZp.forEach {
|
||||||
val varname = asmgen.asmVariableName(it.name)+"_init_value"
|
val varname = asmgen.asmVariableName(it.name)+"_init_value"
|
||||||
arrayVariable2asm(varname, it.alloc.dt, it.value, null)
|
arrayVariable2asm(varname, it.alloc.dt, 0, it.value, null)
|
||||||
}
|
}
|
||||||
|
|
||||||
asmgen.out("""+
|
asmgen.out("+")
|
||||||
clv
|
|
||||||
clc""")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private class ZpStringWithInitial(
|
private class ZpStringWithInitial(
|
||||||
@ -536,24 +557,24 @@ internal class ProgramAndVarsGen(
|
|||||||
|
|
||||||
private fun getZpStringVarsWithInitvalue(): Collection<ZpStringWithInitial> {
|
private fun getZpStringVarsWithInitvalue(): Collection<ZpStringWithInitial> {
|
||||||
val result = mutableListOf<ZpStringWithInitial>()
|
val result = mutableListOf<ZpStringWithInitial>()
|
||||||
val vars = allocator.zeropageVars.filter { it.value.dt==DataType.STR }
|
val vars = allocator.zeropageVars.filter { it.value.dt.isString }
|
||||||
for (variable in vars) {
|
for (variable in vars) {
|
||||||
val scopedName = variable.key
|
val scopedName = variable.key
|
||||||
val svar = symboltable.lookup(scopedName) as? StStaticVariable
|
val svar = symboltable.lookup(scopedName) as? StStaticVariable
|
||||||
if(svar?.onetimeInitializationStringValue!=null)
|
if(svar?.initializationStringValue!=null)
|
||||||
result.add(ZpStringWithInitial(scopedName, variable.value, svar.onetimeInitializationStringValue!!))
|
result.add(ZpStringWithInitial(scopedName, variable.value, svar.initializationStringValue!!))
|
||||||
}
|
}
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun getZpArrayVarsWithInitvalue(): Collection<ZpArrayWithInitial> {
|
private fun getZpArrayVarsWithInitvalue(): Collection<ZpArrayWithInitial> {
|
||||||
val result = mutableListOf<ZpArrayWithInitial>()
|
val result = mutableListOf<ZpArrayWithInitial>()
|
||||||
val vars = allocator.zeropageVars.filter { it.value.dt in ArrayDatatypes }
|
val vars = allocator.zeropageVars.filter { it.value.dt.isArray }
|
||||||
for (variable in vars) {
|
for (variable in vars) {
|
||||||
val scopedName = variable.key
|
val scopedName = variable.key
|
||||||
val svar = symboltable.lookup(scopedName) as? StStaticVariable
|
val svar = symboltable.lookup(scopedName) as? StStaticVariable
|
||||||
if(svar?.onetimeInitializationArrayValue!=null)
|
if(svar?.initializationArrayValue!=null)
|
||||||
result.add(ZpArrayWithInitial(scopedName, variable.value, svar.onetimeInitializationArrayValue!!))
|
result.add(ZpArrayWithInitial(scopedName, variable.value, svar.initializationArrayValue!!))
|
||||||
}
|
}
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
@ -564,7 +585,7 @@ internal class ProgramAndVarsGen(
|
|||||||
if (scopedName.startsWith("cx16.r"))
|
if (scopedName.startsWith("cx16.r"))
|
||||||
continue // The 16 virtual registers of the cx16 are not actual variables in zp, they're memory mapped
|
continue // The 16 virtual registers of the cx16 are not actual variables in zp, they're memory mapped
|
||||||
val variable = symboltable.flat.getValue(scopedName) as StStaticVariable
|
val variable = symboltable.flat.getValue(scopedName) as StStaticVariable
|
||||||
if(variable.dt in SplitWordArrayTypes) {
|
if(variable.dt.isSplitWordArray) {
|
||||||
val lsbAddr = zpvar.address
|
val lsbAddr = zpvar.address
|
||||||
val msbAddr = zpvar.address + (zpvar.size/2).toUInt()
|
val msbAddr = zpvar.address + (zpvar.size/2).toUInt()
|
||||||
asmgen.out("${scopedName.substringAfterLast('.')}_lsb \t= $lsbAddr \t; zp ${zpvar.dt} (lsbs)")
|
asmgen.out("${scopedName.substringAfterLast('.')}_lsb \t= $lsbAddr \t; zp ${zpvar.dt} (lsbs)")
|
||||||
@ -579,43 +600,65 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out("")
|
asmgen.out("")
|
||||||
val (varsNoInit, varsWithInit) = variables.partition { it.uninitialized }
|
val (varsNoInit, varsWithInit) = variables.partition { it.uninitialized }
|
||||||
if(varsNoInit.isNotEmpty()) {
|
if(varsNoInit.isNotEmpty()) {
|
||||||
asmgen.out("; non-zeropage variables without initialization value")
|
asmgen.out("; non-zeropage variables")
|
||||||
asmgen.out(" .section BSS")
|
asmgen.out(" .section BSS")
|
||||||
varsNoInit.sortedWith(compareBy<StStaticVariable> { it.name }.thenBy { it.dt }).forEach {
|
val (notAligned, aligned) = varsNoInit.partition { it.align==0 }
|
||||||
|
notAligned.sortedWith(compareBy<StStaticVariable> { it.name }.thenBy { it.dt.base }).forEach {
|
||||||
|
uninitializedVariable2asm(it)
|
||||||
|
}
|
||||||
|
aligned.sortedWith(compareBy<StStaticVariable> { it.align }.thenBy { it.name }.thenBy { it.dt.base }).forEach {
|
||||||
uninitializedVariable2asm(it)
|
uninitializedVariable2asm(it)
|
||||||
}
|
}
|
||||||
asmgen.out(" .send BSS")
|
asmgen.out(" .send BSS")
|
||||||
}
|
}
|
||||||
|
|
||||||
if(varsWithInit.isNotEmpty()) {
|
if(varsWithInit.isNotEmpty()) {
|
||||||
asmgen.out("; non-zeropage variables")
|
asmgen.out("; non-zeropage variables with init value")
|
||||||
val (stringvars, othervars) = varsWithInit.sortedBy { it.name }.partition { it.dt == DataType.STR }
|
val (stringvars, othervars) = varsWithInit.sortedBy { it.name }.partition { it.dt.isString }
|
||||||
stringvars.forEach {
|
val (notAlignedStrings, alignedStrings) = stringvars.partition { it.align==0 }
|
||||||
|
val (notAlignedOther, alignedOther) = othervars.partition { it.align==0 }
|
||||||
|
notAlignedStrings.forEach {
|
||||||
outputStringvar(
|
outputStringvar(
|
||||||
it.name,
|
it.name,
|
||||||
it.onetimeInitializationStringValue!!.second,
|
it.align,
|
||||||
it.onetimeInitializationStringValue!!.first
|
it.initializationStringValue!!.second,
|
||||||
|
it.initializationStringValue!!.first
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
othervars.sortedBy { it.type }.forEach {
|
alignedStrings.sortedBy { it.align }.forEach {
|
||||||
|
outputStringvar(
|
||||||
|
it.name,
|
||||||
|
it.align,
|
||||||
|
it.initializationStringValue!!.second,
|
||||||
|
it.initializationStringValue!!.first
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
notAlignedOther.sortedBy { it.type }.forEach {
|
||||||
|
staticVariable2asm(it)
|
||||||
|
}
|
||||||
|
alignedOther.sortedBy { it.align }.sortedBy { it.type }.forEach {
|
||||||
staticVariable2asm(it)
|
staticVariable2asm(it)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun uninitializedVariable2asm(variable: StStaticVariable) {
|
private fun uninitializedVariable2asm(variable: StStaticVariable) {
|
||||||
when (variable.dt) {
|
val dt = variable.dt
|
||||||
DataType.BOOL, DataType.UBYTE -> asmgen.out("${variable.name}\t.byte ?")
|
when {
|
||||||
DataType.BYTE -> asmgen.out("${variable.name}\t.char ?")
|
dt.isBool || dt.isUnsignedByte -> asmgen.out("${variable.name}\t.byte ?")
|
||||||
DataType.UWORD -> asmgen.out("${variable.name}\t.word ?")
|
dt.isSignedByte -> asmgen.out("${variable.name}\t.char ?")
|
||||||
DataType.WORD -> asmgen.out("${variable.name}\t.sint ?")
|
dt.isUnsignedWord -> asmgen.out("${variable.name}\t.word ?")
|
||||||
DataType.FLOAT -> asmgen.out("${variable.name}\t.fill ${compTarget.machine.FLOAT_MEM_SIZE}")
|
dt.isSignedWord -> asmgen.out("${variable.name}\t.sint ?")
|
||||||
in SplitWordArrayTypes -> {
|
dt.isFloat -> asmgen.out("${variable.name}\t.fill ${compTarget.machine.FLOAT_MEM_SIZE}")
|
||||||
|
dt.isSplitWordArray -> {
|
||||||
|
alignVar(variable.align)
|
||||||
val numbytesPerHalf = compTarget.memorySize(variable.dt, variable.length!!) / 2
|
val numbytesPerHalf = compTarget.memorySize(variable.dt, variable.length!!) / 2
|
||||||
asmgen.out("${variable.name}_lsb\t.fill $numbytesPerHalf")
|
asmgen.out("${variable.name}_lsb\t.fill $numbytesPerHalf")
|
||||||
asmgen.out("${variable.name}_msb\t.fill $numbytesPerHalf")
|
asmgen.out("${variable.name}_msb\t.fill $numbytesPerHalf")
|
||||||
}
|
}
|
||||||
in ArrayDatatypes -> {
|
dt.isArray -> {
|
||||||
|
alignVar(variable.align)
|
||||||
val numbytes = compTarget.memorySize(variable.dt, variable.length!!)
|
val numbytes = compTarget.memorySize(variable.dt, variable.length!!)
|
||||||
asmgen.out("${variable.name}\t.fill $numbytes")
|
asmgen.out("${variable.name}\t.fill $numbytes")
|
||||||
}
|
}
|
||||||
@ -625,22 +668,27 @@ internal class ProgramAndVarsGen(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun alignVar(align: Int) {
|
||||||
|
if(align > 1)
|
||||||
|
asmgen.out(" .align ${align.toHex()}")
|
||||||
|
}
|
||||||
|
|
||||||
private fun staticVariable2asm(variable: StStaticVariable) {
|
private fun staticVariable2asm(variable: StStaticVariable) {
|
||||||
val initialValue: Number =
|
val initialValue: Number =
|
||||||
if(variable.onetimeInitializationNumericValue!=null) {
|
if(variable.initializationNumericValue!=null) {
|
||||||
if(variable.dt== DataType.FLOAT)
|
if(variable.dt.isFloat)
|
||||||
variable.onetimeInitializationNumericValue!!
|
variable.initializationNumericValue!!
|
||||||
else
|
else
|
||||||
variable.onetimeInitializationNumericValue!!.toInt()
|
variable.initializationNumericValue!!.toInt()
|
||||||
} else 0
|
} else 0
|
||||||
|
|
||||||
when (variable.dt) {
|
val dt=variable.dt
|
||||||
DataType.BOOL -> TODO("bool var to asm")
|
when {
|
||||||
DataType.UBYTE -> asmgen.out("${variable.name}\t.byte ${initialValue.toHex()}")
|
dt.isBool || dt.isUnsignedByte -> asmgen.out("${variable.name}\t.byte ${initialValue.toHex()}")
|
||||||
DataType.BYTE -> asmgen.out("${variable.name}\t.char $initialValue")
|
dt.isSignedByte -> asmgen.out("${variable.name}\t.char $initialValue")
|
||||||
DataType.UWORD -> asmgen.out("${variable.name}\t.word ${initialValue.toHex()}")
|
dt.isUnsignedWord -> asmgen.out("${variable.name}\t.word ${initialValue.toHex()}")
|
||||||
DataType.WORD -> asmgen.out("${variable.name}\t.sint $initialValue")
|
dt.isSignedWord -> asmgen.out("${variable.name}\t.sint $initialValue")
|
||||||
DataType.FLOAT -> {
|
dt.isFloat -> {
|
||||||
if(initialValue==0) {
|
if(initialValue==0) {
|
||||||
asmgen.out("${variable.name}\t.byte 0,0,0,0,0 ; float")
|
asmgen.out("${variable.name}\t.byte 0,0,0,0,0 ; float")
|
||||||
} else {
|
} else {
|
||||||
@ -648,19 +696,22 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out("${variable.name}\t.byte $floatFill ; float $initialValue")
|
asmgen.out("${variable.name}\t.byte $floatFill ; float $initialValue")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DataType.STR -> {
|
dt.isString -> {
|
||||||
throw AssemblyError("all string vars should have been interned into prog")
|
throw AssemblyError("all string vars should have been interned into prog")
|
||||||
}
|
}
|
||||||
in ArrayDatatypes -> arrayVariable2asm(variable.name, variable.dt, variable.onetimeInitializationArrayValue, variable.length)
|
dt.isArray -> {
|
||||||
|
arrayVariable2asm(variable.name, variable.dt, variable.align, variable.initializationArrayValue, variable.length)
|
||||||
|
}
|
||||||
else -> {
|
else -> {
|
||||||
throw AssemblyError("weird dt")
|
throw AssemblyError("weird dt")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun arrayVariable2asm(varname: String, dt: DataType, value: StArray?, orNumberOfZeros: Int?) {
|
private fun arrayVariable2asm(varname: String, dt: DataType, align: Int, value: StArray?, orNumberOfZeros: Int?) {
|
||||||
when(dt) {
|
alignVar(align)
|
||||||
DataType.ARRAY_UB, DataType.ARRAY_BOOL -> {
|
when {
|
||||||
|
dt.isUnsignedByteArray || dt.isBoolArray -> {
|
||||||
val data = makeArrayFillDataUnsigned(dt, value, orNumberOfZeros)
|
val data = makeArrayFillDataUnsigned(dt, value, orNumberOfZeros)
|
||||||
if (data.size <= 16)
|
if (data.size <= 16)
|
||||||
asmgen.out("$varname\t.byte ${data.joinToString()}")
|
asmgen.out("$varname\t.byte ${data.joinToString()}")
|
||||||
@ -670,7 +721,7 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out(" .byte " + chunk.joinToString())
|
asmgen.out(" .byte " + chunk.joinToString())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DataType.ARRAY_B -> {
|
dt.isSignedByteArray -> {
|
||||||
val data = makeArrayFillDataSigned(dt, value, orNumberOfZeros)
|
val data = makeArrayFillDataSigned(dt, value, orNumberOfZeros)
|
||||||
if (data.size <= 16)
|
if (data.size <= 16)
|
||||||
asmgen.out("$varname\t.char ${data.joinToString()}")
|
asmgen.out("$varname\t.char ${data.joinToString()}")
|
||||||
@ -680,7 +731,20 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out(" .char " + chunk.joinToString())
|
asmgen.out(" .char " + chunk.joinToString())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DataType.ARRAY_UW -> {
|
dt.isSplitWordArray -> {
|
||||||
|
if(dt.elementType().isUnsignedWord) {
|
||||||
|
val data = makeArrayFillDataUnsigned(dt, value, orNumberOfZeros)
|
||||||
|
asmgen.out("_array_$varname := ${data.joinToString()}")
|
||||||
|
asmgen.out("${varname}_lsb\t.byte <_array_$varname")
|
||||||
|
asmgen.out("${varname}_msb\t.byte >_array_$varname")
|
||||||
|
} else {
|
||||||
|
val data = makeArrayFillDataSigned(dt, value, orNumberOfZeros)
|
||||||
|
asmgen.out("_array_$varname := ${data.joinToString()}")
|
||||||
|
asmgen.out("${varname}_lsb\t.byte <_array_$varname")
|
||||||
|
asmgen.out("${varname}_msb\t.byte >_array_$varname")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
dt.isUnsignedWordArray -> {
|
||||||
val data = makeArrayFillDataUnsigned(dt, value, orNumberOfZeros)
|
val data = makeArrayFillDataUnsigned(dt, value, orNumberOfZeros)
|
||||||
if (data.size <= 16)
|
if (data.size <= 16)
|
||||||
asmgen.out("$varname\t.word ${data.joinToString()}")
|
asmgen.out("$varname\t.word ${data.joinToString()}")
|
||||||
@ -690,7 +754,7 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out(" .word " + chunk.joinToString())
|
asmgen.out(" .word " + chunk.joinToString())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DataType.ARRAY_W -> {
|
dt.isSignedWordArray -> {
|
||||||
val data = makeArrayFillDataSigned(dt, value, orNumberOfZeros)
|
val data = makeArrayFillDataSigned(dt, value, orNumberOfZeros)
|
||||||
if (data.size <= 16)
|
if (data.size <= 16)
|
||||||
asmgen.out("$varname\t.sint ${data.joinToString()}")
|
asmgen.out("$varname\t.sint ${data.joinToString()}")
|
||||||
@ -700,19 +764,7 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out(" .sint " + chunk.joinToString())
|
asmgen.out(" .sint " + chunk.joinToString())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DataType.ARRAY_UW_SPLIT -> {
|
dt.isFloatArray -> {
|
||||||
val data = makeArrayFillDataUnsigned(dt, value, orNumberOfZeros)
|
|
||||||
asmgen.out("_array_$varname := ${data.joinToString()}")
|
|
||||||
asmgen.out("${varname}_lsb\t.byte <_array_$varname")
|
|
||||||
asmgen.out("${varname}_msb\t.byte >_array_$varname")
|
|
||||||
}
|
|
||||||
DataType.ARRAY_W_SPLIT -> {
|
|
||||||
val data = makeArrayFillDataSigned(dt, value, orNumberOfZeros)
|
|
||||||
asmgen.out("_array_$varname := ${data.joinToString()}")
|
|
||||||
asmgen.out("${varname}_lsb\t.byte <_array_$varname")
|
|
||||||
asmgen.out("${varname}_msb\t.byte >_array_$varname")
|
|
||||||
}
|
|
||||||
DataType.ARRAY_F -> {
|
|
||||||
val array = value ?: zeroFilledArray(orNumberOfZeros!!)
|
val array = value ?: zeroFilledArray(orNumberOfZeros!!)
|
||||||
val floatFills = array.map {
|
val floatFills = array.map {
|
||||||
compTarget.machine.getFloatAsmBytes(it.number!!)
|
compTarget.machine.getFloatAsmBytes(it.number!!)
|
||||||
@ -738,7 +790,7 @@ internal class ProgramAndVarsGen(
|
|||||||
asmgen.out(" ${it.name} = ${it.address.toHex()}")
|
asmgen.out(" ${it.name} = ${it.address.toHex()}")
|
||||||
}
|
}
|
||||||
consts.sortedBy { it.name }.forEach {
|
consts.sortedBy { it.name }.forEach {
|
||||||
if(it.dt==DataType.FLOAT)
|
if(it.dt==BaseDataType.FLOAT)
|
||||||
asmgen.out(" ${it.name} = ${it.value}")
|
asmgen.out(" ${it.name} = ${it.value}")
|
||||||
else
|
else
|
||||||
asmgen.out(" ${it.name} = ${it.value.toHex()}")
|
asmgen.out(" ${it.name} = ${it.value.toHex()}")
|
||||||
@ -750,11 +802,16 @@ internal class ProgramAndVarsGen(
|
|||||||
.filter { it is PtAsmSub && it.address!=null }
|
.filter { it is PtAsmSub && it.address!=null }
|
||||||
.forEach { asmsub ->
|
.forEach { asmsub ->
|
||||||
asmsub as PtAsmSub
|
asmsub as PtAsmSub
|
||||||
asmgen.out(" ${asmsub.name} = ${asmsub.address!!.toHex()}")
|
val address = asmsub.address!!
|
||||||
|
val bank = if(address.constbank!=null) "; @bank ${address.constbank}"
|
||||||
|
else if(address.varbank!=null) "; @bank ${address.varbank?.name}"
|
||||||
|
else ""
|
||||||
|
asmgen.out(" ${asmsub.name} = ${address.address.toHex()} $bank")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun outputStringvar(varname: String, encoding: Encoding, value: String) {
|
private fun outputStringvar(varname: String, align: Int, encoding: Encoding, value: String) {
|
||||||
|
alignVar(align)
|
||||||
asmgen.out("$varname\t; $encoding:\"${value.escape().replace("\u0000", "<NULL>")}\"", false)
|
asmgen.out("$varname\t; $encoding:\"${value.escape().replace("\u0000", "<NULL>")}\"", false)
|
||||||
val bytes = compTarget.encodeString(value, encoding).plus(0.toUByte())
|
val bytes = compTarget.encodeString(value, encoding).plus(0.toUByte())
|
||||||
val outputBytes = bytes.map { "$" + it.toString(16).padStart(2, '0') }
|
val outputBytes = bytes.map { "$" + it.toString(16).padStart(2, '0') }
|
||||||
@ -764,8 +821,8 @@ internal class ProgramAndVarsGen(
|
|||||||
|
|
||||||
private fun makeArrayFillDataUnsigned(dt: DataType, value: StArray?, orNumberOfZeros: Int?): List<String> {
|
private fun makeArrayFillDataUnsigned(dt: DataType, value: StArray?, orNumberOfZeros: Int?): List<String> {
|
||||||
val array = value ?: zeroFilledArray(orNumberOfZeros!!)
|
val array = value ?: zeroFilledArray(orNumberOfZeros!!)
|
||||||
return when (dt) {
|
return when {
|
||||||
DataType.ARRAY_BOOL ->
|
dt.isBoolArray ->
|
||||||
// byte array can never contain pointer-to types, so treat values as all integers
|
// byte array can never contain pointer-to types, so treat values as all integers
|
||||||
array.map {
|
array.map {
|
||||||
if(it.boolean!=null)
|
if(it.boolean!=null)
|
||||||
@ -775,18 +832,23 @@ internal class ProgramAndVarsGen(
|
|||||||
if(number==0.0) "0" else "1"
|
if(number==0.0) "0" else "1"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DataType.ARRAY_UB ->
|
dt.isUnsignedByteArray ->
|
||||||
// byte array can never contain pointer-to types, so treat values as all integers
|
// byte array can never contain pointer-to types, so treat values as all integers
|
||||||
array.map {
|
array.map {
|
||||||
val number = it.number!!.toInt()
|
val number = it.number!!.toInt()
|
||||||
"$"+number.toString(16).padStart(2, '0')
|
"$"+number.toString(16).padStart(2, '0')
|
||||||
}
|
}
|
||||||
DataType.ARRAY_UW, DataType.ARRAY_UW_SPLIT -> array.map {
|
dt.isArray && dt.elementType().isUnsignedWord -> array.map {
|
||||||
if(it.number!=null) {
|
if(it.number!=null) {
|
||||||
"$" + it.number!!.toInt().toString(16).padStart(4, '0')
|
"$" + it.number!!.toInt().toString(16).padStart(4, '0')
|
||||||
}
|
}
|
||||||
else if(it.addressOfSymbol!=null) {
|
else if(it.addressOfSymbol!=null) {
|
||||||
asmgen.asmSymbolName(it.addressOfSymbol!!)
|
val addrOfSymbol = it.addressOfSymbol!!
|
||||||
|
val symbol = symboltable.lookup(addrOfSymbol)!!
|
||||||
|
if(symbol is StStaticVariable && symbol.dt.isSplitWordArray)
|
||||||
|
asmgen.asmSymbolName(addrOfSymbol+"_lsb") // the _lsb split array comes first in memory
|
||||||
|
else
|
||||||
|
asmgen.asmSymbolName(addrOfSymbol)
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
throw AssemblyError("weird array elt")
|
throw AssemblyError("weird array elt")
|
||||||
@ -797,14 +859,14 @@ internal class ProgramAndVarsGen(
|
|||||||
|
|
||||||
private fun makeArrayFillDataSigned(dt: DataType, value: StArray?, orNumberOfZeros: Int?): List<String> {
|
private fun makeArrayFillDataSigned(dt: DataType, value: StArray?, orNumberOfZeros: Int?): List<String> {
|
||||||
val array = value ?: zeroFilledArray(orNumberOfZeros!!)
|
val array = value ?: zeroFilledArray(orNumberOfZeros!!)
|
||||||
return when (dt) {
|
return when {
|
||||||
// byte array can never contain pointer-to types, so treat values as all integers
|
// byte array can never contain pointer-to types, so treat values as all integers
|
||||||
DataType.ARRAY_UB ->
|
dt.isUnsignedByteArray ->
|
||||||
array.map {
|
array.map {
|
||||||
val number = it.number!!.toInt()
|
val number = it.number!!.toInt()
|
||||||
"$"+number.toString(16).padStart(2, '0')
|
"$"+number.toString(16).padStart(2, '0')
|
||||||
}
|
}
|
||||||
DataType.ARRAY_B ->
|
dt.isSignedByteArray ->
|
||||||
array.map {
|
array.map {
|
||||||
val number = it.number!!.toInt()
|
val number = it.number!!.toInt()
|
||||||
val hexnum = number.absoluteValue.toString(16).padStart(2, '0')
|
val hexnum = number.absoluteValue.toString(16).padStart(2, '0')
|
||||||
@ -813,11 +875,11 @@ internal class ProgramAndVarsGen(
|
|||||||
else
|
else
|
||||||
"-$$hexnum"
|
"-$$hexnum"
|
||||||
}
|
}
|
||||||
DataType.ARRAY_UW, DataType.ARRAY_UW_SPLIT -> array.map {
|
dt.isArray && dt.elementType().isUnsignedWord -> array.map {
|
||||||
val number = it.number!!.toInt()
|
val number = it.number!!.toInt()
|
||||||
"$" + number.toString(16).padStart(4, '0')
|
"$" + number.toString(16).padStart(4, '0')
|
||||||
}
|
}
|
||||||
DataType.ARRAY_W, DataType.ARRAY_W_SPLIT -> array.map {
|
dt.isArray && dt.elementType().isSignedWord -> array.map {
|
||||||
val number = it.number!!.toInt()
|
val number = it.number!!.toInt()
|
||||||
val hexnum = number.absoluteValue.toString(16).padStart(4, '0')
|
val hexnum = number.absoluteValue.toString(16).padStart(4, '0')
|
||||||
if(number>=0)
|
if(number>=0)
|
||||||
|
@ -48,9 +48,9 @@ internal class VariableAllocator(private val symboltable: SymbolTable,
|
|||||||
val numberOfAllocatableVariables = allVariables.size
|
val numberOfAllocatableVariables = allVariables.size
|
||||||
val varsRequiringZp = allVariables.filter { it.zpwish == ZeropageWish.REQUIRE_ZEROPAGE }
|
val varsRequiringZp = allVariables.filter { it.zpwish == ZeropageWish.REQUIRE_ZEROPAGE }
|
||||||
val varsPreferringZp = allVariables.filter { it.zpwish == ZeropageWish.PREFER_ZEROPAGE }
|
val varsPreferringZp = allVariables.filter { it.zpwish == ZeropageWish.PREFER_ZEROPAGE }
|
||||||
val varsDontCare = allVariables.filter { it.zpwish == ZeropageWish.DONTCARE }
|
val varsNotZp = allVariables.filter { it.zpwish == ZeropageWish.NOT_IN_ZEROPAGE }
|
||||||
val numberOfExplicitNonZpVariables = allVariables.count { it.zpwish == ZeropageWish.NOT_IN_ZEROPAGE }
|
val (varsDontCareWithoutAlignment, varsDontCareWithAlignment) = allVariables.filter { it.zpwish == ZeropageWish.DONTCARE }.partition { it.align == 0 }
|
||||||
require(varsDontCare.size + varsRequiringZp.size + varsPreferringZp.size + numberOfExplicitNonZpVariables == numberOfAllocatableVariables)
|
require(varsDontCareWithAlignment.size + varsDontCareWithoutAlignment.size + varsRequiringZp.size + varsPreferringZp.size + varsNotZp.size == numberOfAllocatableVariables)
|
||||||
|
|
||||||
var numVariablesAllocatedInZP = 0
|
var numVariablesAllocatedInZP = 0
|
||||||
var numberOfNonIntegerVariables = 0
|
var numberOfNonIntegerVariables = 0
|
||||||
@ -60,7 +60,7 @@ internal class VariableAllocator(private val symboltable: SymbolTable,
|
|||||||
variable.scopedName,
|
variable.scopedName,
|
||||||
variable.dt,
|
variable.dt,
|
||||||
variable.length,
|
variable.length,
|
||||||
variable.astNode.position,
|
variable.astNode?.position ?: Position.DUMMY,
|
||||||
errors
|
errors
|
||||||
)
|
)
|
||||||
result.fold(
|
result.fold(
|
||||||
@ -68,7 +68,7 @@ internal class VariableAllocator(private val symboltable: SymbolTable,
|
|||||||
numVariablesAllocatedInZP++
|
numVariablesAllocatedInZP++
|
||||||
},
|
},
|
||||||
failure = {
|
failure = {
|
||||||
errors.err(it.message!!, variable.astNode.position)
|
errors.err(it.message!!, variable.astNode?.position ?: Position.DUMMY)
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -79,19 +79,19 @@ internal class VariableAllocator(private val symboltable: SymbolTable,
|
|||||||
variable.scopedName,
|
variable.scopedName,
|
||||||
variable.dt,
|
variable.dt,
|
||||||
variable.length,
|
variable.length,
|
||||||
variable.astNode.position,
|
variable.astNode?.position ?: Position.DUMMY,
|
||||||
errors
|
errors
|
||||||
)
|
)
|
||||||
result.onSuccess { numVariablesAllocatedInZP++ }
|
result.onSuccess { numVariablesAllocatedInZP++ }
|
||||||
// no need to check for allocation error, if there is one, just allocate in normal system ram.
|
// no need to check for allocation error, if there is one, just allocate in normal system ram.
|
||||||
}
|
}
|
||||||
|
|
||||||
// try to allocate any other interger variables into the zeropage until it is full.
|
// try to allocate the "don't care" interger variables into the zeropage until it is full.
|
||||||
// TODO some form of intelligent priorization? most often used variables first? loopcounter vars first? ...?
|
// TODO some form of intelligent priorization? most often used variables first? loopcounter vars first? ...?
|
||||||
if(errors.noErrors()) {
|
if(errors.noErrors()) {
|
||||||
val sortedList = varsDontCare.sortedByDescending { it.scopedName }
|
val sortedList = varsDontCareWithoutAlignment.sortedByDescending { it.scopedName }
|
||||||
for (variable in sortedList) {
|
for (variable in sortedList) {
|
||||||
if(variable.dt in IntegerDatatypesWithBoolean) {
|
if(variable.dt.isIntegerOrBool) {
|
||||||
if(zeropage.free.isEmpty()) {
|
if(zeropage.free.isEmpty()) {
|
||||||
break
|
break
|
||||||
} else {
|
} else {
|
||||||
@ -99,7 +99,7 @@ internal class VariableAllocator(private val symboltable: SymbolTable,
|
|||||||
variable.scopedName,
|
variable.scopedName,
|
||||||
variable.dt,
|
variable.dt,
|
||||||
variable.length,
|
variable.length,
|
||||||
variable.astNode.position,
|
variable.astNode?.position ?: Position.DUMMY,
|
||||||
errors
|
errors
|
||||||
)
|
)
|
||||||
result.onSuccess { numVariablesAllocatedInZP++ }
|
result.onSuccess { numVariablesAllocatedInZP++ }
|
||||||
@ -110,9 +110,7 @@ internal class VariableAllocator(private val symboltable: SymbolTable,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// println(" number of allocated vars: $numberOfAllocatableVariables")
|
// note: no zeropage allocation is done at all for the @nozp variables. This means they will always end up outside the zeropage.
|
||||||
// println(" put into zeropage: $numVariablesAllocatedInZP, non-zp allocatable: ${numberOfNonIntegerVariables+numberOfExplicitNonZpVariables}")
|
|
||||||
// println(" zeropage free space: ${zeropage.free.size} bytes")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun collectAllVariables(st: SymbolTable): Collection<StStaticVariable> {
|
private fun collectAllVariables(st: SymbolTable): Collection<StStaticVariable> {
|
||||||
@ -126,6 +124,6 @@ internal class VariableAllocator(private val symboltable: SymbolTable,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
collect(st)
|
collect(st)
|
||||||
return vars.sortedBy { it.dt }
|
return vars.sortedBy { it.dt.base }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -17,28 +17,28 @@ internal class AnyExprAsmGen(
|
|||||||
private val asmgen: AsmGen6502Internal
|
private val asmgen: AsmGen6502Internal
|
||||||
) {
|
) {
|
||||||
fun assignAnyExpressionUsingStack(expr: PtBinaryExpression, assign: AsmAssignment): Boolean {
|
fun assignAnyExpressionUsingStack(expr: PtBinaryExpression, assign: AsmAssignment): Boolean {
|
||||||
when(expr.type) {
|
when {
|
||||||
in ByteDatatypesWithBoolean -> {
|
expr.type.isByteOrBool -> {
|
||||||
if(expr.left.type in ByteDatatypesWithBoolean && expr.right.type in ByteDatatypesWithBoolean)
|
if(expr.left.type.isByteOrBool && expr.right.type.isByteOrBool)
|
||||||
return assignByteBinExpr(expr, assign)
|
return assignByteBinExpr(expr, assign)
|
||||||
if (expr.left.type in WordDatatypes && expr.right.type in WordDatatypes) {
|
if (expr.left.type.isWord && expr.right.type.isWord) {
|
||||||
require(expr.operator in ComparisonOperators)
|
require(expr.operator in ComparisonOperators)
|
||||||
throw AssemblyError("words operands comparison -> byte, should have been handled elsewhere")
|
throw AssemblyError("words operands comparison -> byte, should have been handled elsewhere")
|
||||||
}
|
}
|
||||||
if (expr.left.type==DataType.FLOAT && expr.right.type==DataType.FLOAT) {
|
if (expr.left.type.isFloat && expr.right.type.isFloat) {
|
||||||
require(expr.operator in ComparisonOperators)
|
require(expr.operator in ComparisonOperators)
|
||||||
return assignFloatBinExpr(expr, assign)
|
return assignFloatBinExpr(expr, assign)
|
||||||
}
|
}
|
||||||
throw AssemblyError("weird expr operand types: ${expr.left.type} and ${expr.right.type}")
|
throw AssemblyError("weird expr operand types: ${expr.left.type} and ${expr.right.type}")
|
||||||
}
|
}
|
||||||
in WordDatatypes -> {
|
expr.type.isWord -> {
|
||||||
require(expr.left.type in WordDatatypes && expr.right.type in WordDatatypes) {
|
require(expr.left.type.isWord && expr.right.type.isWord) {
|
||||||
"both operands must be words"
|
"both operands must be words"
|
||||||
}
|
}
|
||||||
return assignWordBinExpr(expr)
|
throw AssemblyError("expression should have been handled otherwise: word ${expr.operator} at ${expr.position}")
|
||||||
}
|
}
|
||||||
DataType.FLOAT -> {
|
expr.type.isFloat -> {
|
||||||
require(expr.left.type==DataType.FLOAT && expr.right.type==DataType.FLOAT) {
|
require(expr.left.type.isFloat && expr.right.type.isFloat) {
|
||||||
"both operands must be floats"
|
"both operands must be floats"
|
||||||
}
|
}
|
||||||
return assignFloatBinExpr(expr, assign)
|
return assignFloatBinExpr(expr, assign)
|
||||||
@ -47,36 +47,12 @@ internal class AnyExprAsmGen(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun assignWordBinExpr(expr: PtBinaryExpression): Boolean {
|
|
||||||
when(expr.operator) {
|
|
||||||
"+" -> TODO("word + at ${expr.position}")
|
|
||||||
"-" -> TODO("word - at ${expr.position}")
|
|
||||||
"*" -> TODO("word * at ${expr.position}")
|
|
||||||
"/" -> TODO("word / at ${expr.position}")
|
|
||||||
"<<" -> TODO("word << at ${expr.position}")
|
|
||||||
">>" -> TODO("word >> at ${expr.position}")
|
|
||||||
"%" -> TODO("word % at ${expr.position}")
|
|
||||||
"and" -> TODO("word logical and (with optional shortcircuit) ${expr.position}")
|
|
||||||
"or" -> TODO("word logical or (with optional shortcircuit) ${expr.position}")
|
|
||||||
"&" -> TODO("word and at ${expr.position}")
|
|
||||||
"|" -> TODO("word or at ${expr.position}")
|
|
||||||
"^", "xor" -> TODO("word xor at ${expr.position}")
|
|
||||||
"==" -> TODO("word == at ${expr.position}")
|
|
||||||
"!=" -> TODO("word != at ${expr.position}")
|
|
||||||
"<" -> TODO("word < at ${expr.position}")
|
|
||||||
"<=" -> TODO("word <= at ${expr.position}")
|
|
||||||
">" -> TODO("word > at ${expr.position}")
|
|
||||||
">=" -> TODO("word >= at ${expr.position}")
|
|
||||||
else -> return false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun assignByteBinExpr(expr: PtBinaryExpression, assign: AsmAssignment): Boolean {
|
private fun assignByteBinExpr(expr: PtBinaryExpression, assign: AsmAssignment): Boolean {
|
||||||
when(expr.operator) {
|
when(expr.operator) {
|
||||||
"+" -> {
|
"+" -> {
|
||||||
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
||||||
asmgen.out(" pha")
|
asmgen.out(" pha")
|
||||||
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE)
|
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.forDt(BaseDataType.UBYTE))
|
||||||
asmgen.out(" pla | clc | adc P8ZP_SCRATCH_B1")
|
asmgen.out(" pla | clc | adc P8ZP_SCRATCH_B1")
|
||||||
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
||||||
return true
|
return true
|
||||||
@ -84,32 +60,22 @@ internal class AnyExprAsmGen(
|
|||||||
"-" -> {
|
"-" -> {
|
||||||
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
||||||
asmgen.out(" pha")
|
asmgen.out(" pha")
|
||||||
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE)
|
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.forDt(BaseDataType.UBYTE))
|
||||||
asmgen.out(" pla | sec | sbc P8ZP_SCRATCH_B1")
|
asmgen.out(" pla | sec | sbc P8ZP_SCRATCH_B1")
|
||||||
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
"*" -> {
|
"*" -> TODO("byte * at ${expr.position}")
|
||||||
TODO("byte * at ${expr.position}")
|
"/" -> TODO("byte / at ${expr.position}")
|
||||||
}
|
"<<" -> TODO("byte << at ${expr.position}")
|
||||||
"/" -> {
|
">>" -> TODO("byte >> at ${expr.position}")
|
||||||
TODO("byte / at ${expr.position}")
|
"%" -> TODO("byte % at ${expr.position}")
|
||||||
}
|
|
||||||
"<<" -> {
|
|
||||||
TODO("byte << at ${expr.position}")
|
|
||||||
}
|
|
||||||
">>" -> {
|
|
||||||
TODO("byte >> at ${expr.position}")
|
|
||||||
}
|
|
||||||
"%" -> {
|
|
||||||
TODO("byte % at ${expr.position}")
|
|
||||||
}
|
|
||||||
"and" -> TODO("logical and (with optional shortcircuit) ${expr.position}")
|
"and" -> TODO("logical and (with optional shortcircuit) ${expr.position}")
|
||||||
"or" -> TODO("logical or (with optional shortcircuit) ${expr.position}")
|
"or" -> TODO("logical or (with optional shortcircuit) ${expr.position}")
|
||||||
"&" -> {
|
"&" -> {
|
||||||
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
||||||
asmgen.out(" pha")
|
asmgen.out(" pha")
|
||||||
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE)
|
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.forDt(BaseDataType.UBYTE))
|
||||||
asmgen.out(" pla | and P8ZP_SCRATCH_B1")
|
asmgen.out(" pla | and P8ZP_SCRATCH_B1")
|
||||||
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
||||||
return true
|
return true
|
||||||
@ -117,7 +83,7 @@ internal class AnyExprAsmGen(
|
|||||||
"|" -> {
|
"|" -> {
|
||||||
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
||||||
asmgen.out(" pha")
|
asmgen.out(" pha")
|
||||||
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE)
|
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.forDt(BaseDataType.UBYTE))
|
||||||
asmgen.out(" pla | ora P8ZP_SCRATCH_B1")
|
asmgen.out(" pla | ora P8ZP_SCRATCH_B1")
|
||||||
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
||||||
return true
|
return true
|
||||||
@ -125,29 +91,17 @@ internal class AnyExprAsmGen(
|
|||||||
"^", "xor" -> {
|
"^", "xor" -> {
|
||||||
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A, false)
|
||||||
asmgen.out(" pha")
|
asmgen.out(" pha")
|
||||||
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE)
|
asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.forDt(BaseDataType.UBYTE))
|
||||||
asmgen.out(" pla | eor P8ZP_SCRATCH_B1")
|
asmgen.out(" pla | eor P8ZP_SCRATCH_B1")
|
||||||
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
asmgen.assignRegister(RegisterOrPair.A, assign.target)
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
"==" -> {
|
"==" -> TODO("byte == at ${expr.position}")
|
||||||
TODO("byte == at ${expr.position}")
|
"!=" -> TODO("byte != at ${expr.position}")
|
||||||
}
|
"<" -> TODO("byte < at ${expr.position}")
|
||||||
"!=" -> {
|
"<=" -> TODO("byte <= at ${expr.position}")
|
||||||
TODO("byte != at ${expr.position}")
|
">" -> TODO("byte > at ${expr.position}")
|
||||||
}
|
">=" -> TODO("byte >= at ${expr.position}")
|
||||||
"<" -> {
|
|
||||||
TODO("byte < at ${expr.position}")
|
|
||||||
}
|
|
||||||
"<=" -> {
|
|
||||||
TODO("byte <= at ${expr.position}")
|
|
||||||
}
|
|
||||||
">" -> {
|
|
||||||
TODO("byte > at ${expr.position}")
|
|
||||||
}
|
|
||||||
">=" -> {
|
|
||||||
TODO("byte >= at ${expr.position}")
|
|
||||||
}
|
|
||||||
else -> return false
|
else -> return false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -221,17 +175,14 @@ internal class AnyExprAsmGen(
|
|||||||
private fun assignFloatOperandsToFACandARG(left: PtExpression, right: PtExpression) {
|
private fun assignFloatOperandsToFACandARG(left: PtExpression, right: PtExpression) {
|
||||||
when(asmgen.options.compTarget.name) {
|
when(asmgen.options.compTarget.name) {
|
||||||
C64Target.NAME -> {
|
C64Target.NAME -> {
|
||||||
// c64 has a quirk: always make sure FAC2 is loaded last (done using CONUPK) otherwise the result will be corrupt on C64
|
// C64 math library has a quirk: you have always make sure FAC2/ARG is loaded last (done using CONUPK)
|
||||||
// this requires some more forced copying around of float values in certain cases
|
// otherwise the result of certain floating point operations such as FDIVT will be wrong.
|
||||||
if (right.isSimple()) {
|
// see https://www.c64-wiki.com/wiki/CONUPK
|
||||||
asmgen.assignExpressionToRegister(left, RegisterOrPair.FAC1, true)
|
// Unfortunately this means we have to push and pop an intermediary floating point value to and from memory.
|
||||||
asmgen.assignExpressionToRegister(right, RegisterOrPair.FAC2, true)
|
asmgen.assignExpressionToRegister(right, RegisterOrPair.FAC1, true)
|
||||||
} else {
|
asmgen.pushFAC1()
|
||||||
asmgen.assignExpressionToRegister(right, RegisterOrPair.FAC1, true)
|
asmgen.assignExpressionToRegister(left, RegisterOrPair.FAC1, true)
|
||||||
asmgen.pushFAC1()
|
asmgen.popFAC2()
|
||||||
asmgen.assignExpressionToRegister(left, RegisterOrPair.FAC1, true)
|
|
||||||
asmgen.popFAC2()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
Cx16Target.NAME -> {
|
Cx16Target.NAME -> {
|
||||||
asmgen.assignExpressionToRegister(left, RegisterOrPair.FAC1, true)
|
asmgen.assignExpressionToRegister(left, RegisterOrPair.FAC1, true)
|
||||||
@ -246,7 +197,7 @@ internal class AnyExprAsmGen(
|
|||||||
private fun setupFloatComparisonFAC1vsVarAY(expr: PtBinaryExpression) {
|
private fun setupFloatComparisonFAC1vsVarAY(expr: PtBinaryExpression) {
|
||||||
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.FAC1, true)
|
asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.FAC1, true)
|
||||||
if(!expr.right.isSimple()) asmgen.pushFAC1()
|
if(!expr.right.isSimple()) asmgen.pushFAC1()
|
||||||
asmgen.assignExpressionToVariable(expr.right, "floats.floats_temp_var", DataType.FLOAT)
|
asmgen.assignExpressionToVariable(expr.right, "floats.floats_temp_var", DataType.forDt(BaseDataType.FLOAT))
|
||||||
if(!expr.right.isSimple()) asmgen.popFAC1()
|
if(!expr.right.isSimple()) asmgen.popFAC1()
|
||||||
asmgen.out(" lda #<floats.floats_temp_var | ldy #>floats.floats_temp_var")
|
asmgen.out(" lda #<floats.floats_temp_var | ldy #>floats.floats_temp_var")
|
||||||
}
|
}
|
||||||
|
@ -44,7 +44,7 @@ internal class AsmAssignTarget(val kind: TargetStorageKind,
|
|||||||
}
|
}
|
||||||
|
|
||||||
init {
|
init {
|
||||||
if(register!=null && datatype !in NumericDatatypesWithBoolean)
|
if(register!=null && !datatype.isNumericOrBool)
|
||||||
throw AssemblyError("must be numeric type")
|
throw AssemblyError("must be numeric type")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -77,12 +77,20 @@ internal class AsmAssignTarget(val kind: TargetStorageKind,
|
|||||||
when(registers) {
|
when(registers) {
|
||||||
RegisterOrPair.A,
|
RegisterOrPair.A,
|
||||||
RegisterOrPair.X,
|
RegisterOrPair.X,
|
||||||
RegisterOrPair.Y -> AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, if(signed) DataType.BYTE else DataType.UBYTE, scope, pos, register = registers)
|
RegisterOrPair.Y -> {
|
||||||
|
val dt = DataType.forDt(if(signed) BaseDataType.BYTE else BaseDataType.UBYTE)
|
||||||
|
AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, dt, scope, pos, register = registers)
|
||||||
|
}
|
||||||
RegisterOrPair.AX,
|
RegisterOrPair.AX,
|
||||||
RegisterOrPair.AY,
|
RegisterOrPair.AY,
|
||||||
RegisterOrPair.XY -> AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, if(signed) DataType.WORD else DataType.UWORD, scope, pos, register = registers)
|
RegisterOrPair.XY -> {
|
||||||
|
val dt = DataType.forDt(if(signed) BaseDataType.WORD else BaseDataType.UWORD)
|
||||||
|
AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, dt, scope, pos, register = registers)
|
||||||
|
}
|
||||||
RegisterOrPair.FAC1,
|
RegisterOrPair.FAC1,
|
||||||
RegisterOrPair.FAC2 -> AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, DataType.FLOAT, scope, pos, register = registers)
|
RegisterOrPair.FAC2 -> {
|
||||||
|
AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, DataType.forDt(BaseDataType.FLOAT), scope, pos, register = registers)
|
||||||
|
}
|
||||||
RegisterOrPair.R0,
|
RegisterOrPair.R0,
|
||||||
RegisterOrPair.R1,
|
RegisterOrPair.R1,
|
||||||
RegisterOrPair.R2,
|
RegisterOrPair.R2,
|
||||||
@ -98,7 +106,10 @@ internal class AsmAssignTarget(val kind: TargetStorageKind,
|
|||||||
RegisterOrPair.R12,
|
RegisterOrPair.R12,
|
||||||
RegisterOrPair.R13,
|
RegisterOrPair.R13,
|
||||||
RegisterOrPair.R14,
|
RegisterOrPair.R14,
|
||||||
RegisterOrPair.R15 -> AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, if(signed) DataType.WORD else DataType.UWORD, scope, pos, register = registers)
|
RegisterOrPair.R15 -> {
|
||||||
|
val dt = DataType.forDt(if(signed) BaseDataType.WORD else BaseDataType.UWORD)
|
||||||
|
AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, dt, scope, pos, register = registers)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -151,7 +162,7 @@ internal class AsmAssignSource(val kind: SourceStorageKind,
|
|||||||
return AsmAssignSource(SourceStorageKind.LITERALNUMBER, program, asmgen, cv.type, number = cv)
|
return AsmAssignSource(SourceStorageKind.LITERALNUMBER, program, asmgen, cv.type, number = cv)
|
||||||
val bv = value as? PtBool
|
val bv = value as? PtBool
|
||||||
if(bv!=null)
|
if(bv!=null)
|
||||||
return AsmAssignSource(SourceStorageKind.LITERALBOOLEAN, program, asmgen, DataType.BOOL, boolean = bv)
|
return AsmAssignSource(SourceStorageKind.LITERALBOOLEAN, program, asmgen, DataType.forDt(BaseDataType.BOOL), boolean = bv)
|
||||||
|
|
||||||
return when(value) {
|
return when(value) {
|
||||||
// checked above: is PtNumber -> throw AssemblyError("should have been constant value")
|
// checked above: is PtNumber -> throw AssemblyError("should have been constant value")
|
||||||
@ -163,7 +174,7 @@ internal class AsmAssignSource(val kind: SourceStorageKind,
|
|||||||
throw AssemblyError("can't assign from a asmsub register parameter $value ${value.position}")
|
throw AssemblyError("can't assign from a asmsub register parameter $value ${value.position}")
|
||||||
val varName=asmgen.asmVariableName(value)
|
val varName=asmgen.asmVariableName(value)
|
||||||
// special case: "cx16.r[0-15]" are 16-bits virtual registers of the commander X16 system
|
// special case: "cx16.r[0-15]" are 16-bits virtual registers of the commander X16 system
|
||||||
if(value.type == DataType.UWORD && varName.lowercase().startsWith("cx16.r")) {
|
if(value.type.isUnsignedWord && varName.lowercase().startsWith("cx16.r")) {
|
||||||
val regStr = varName.lowercase().substring(5)
|
val regStr = varName.lowercase().substring(5)
|
||||||
val reg = RegisterOrPair.valueOf(regStr.uppercase())
|
val reg = RegisterOrPair.valueOf(regStr.uppercase())
|
||||||
AsmAssignSource(SourceStorageKind.REGISTER, program, asmgen, value.type, register = reg)
|
AsmAssignSource(SourceStorageKind.REGISTER, program, asmgen, value.type, register = reg)
|
||||||
@ -172,7 +183,7 @@ internal class AsmAssignSource(val kind: SourceStorageKind,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
is PtMemoryByte -> {
|
is PtMemoryByte -> {
|
||||||
AsmAssignSource(SourceStorageKind.MEMORY, program, asmgen, DataType.UBYTE, memory = value)
|
AsmAssignSource(SourceStorageKind.MEMORY, program, asmgen, DataType.forDt(BaseDataType.UBYTE), memory = value)
|
||||||
}
|
}
|
||||||
is PtArrayIndexer -> {
|
is PtArrayIndexer -> {
|
||||||
AsmAssignSource(SourceStorageKind.ARRAY, program, asmgen, value.type, array = value)
|
AsmAssignSource(SourceStorageKind.ARRAY, program, asmgen, value.type, array = value)
|
||||||
@ -202,9 +213,9 @@ internal class AsmAssignSource(val kind: SourceStorageKind,
|
|||||||
AsmAssignSource(kind, program, asmgen, newType, variableAsmName, array, memory, register, number, boolean, expression)
|
AsmAssignSource(kind, program, asmgen, newType, variableAsmName, array, memory, register, number, boolean, expression)
|
||||||
|
|
||||||
if(target.datatype!=datatype) {
|
if(target.datatype!=datatype) {
|
||||||
if(target.datatype in ByteDatatypes && datatype in ByteDatatypes) {
|
if(target.datatype.isByte && datatype.isByte) {
|
||||||
return withAdjustedDt(target.datatype)
|
return withAdjustedDt(target.datatype)
|
||||||
} else if(target.datatype in WordDatatypes && datatype in WordDatatypes) {
|
} else if(target.datatype.isWord && datatype.isWord) {
|
||||||
return withAdjustedDt(target.datatype)
|
return withAdjustedDt(target.datatype)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -220,8 +231,9 @@ internal sealed class AsmAssignmentBase(val source: AsmAssignSource,
|
|||||||
val position: Position) {
|
val position: Position) {
|
||||||
init {
|
init {
|
||||||
if(target.register !in arrayOf(RegisterOrPair.XY, RegisterOrPair.AX, RegisterOrPair.AY))
|
if(target.register !in arrayOf(RegisterOrPair.XY, RegisterOrPair.AX, RegisterOrPair.AY))
|
||||||
require(source.datatype != DataType.UNDEFINED) { "must not be placeholder/undefined datatype at $position" }
|
require(!source.datatype.isUndefined) { "must not be placeholder/undefined datatype at $position" }
|
||||||
require(memsizer.memorySize(source.datatype) <= memsizer.memorySize(target.datatype)) {
|
if(!source.datatype.isArray && !target.datatype.isArray)
|
||||||
|
require(memsizer.memorySize(source.datatype, null) <= memsizer.memorySize(target.datatype, null)) {
|
||||||
"source dt size must be less or equal to target dt size at $position srcdt=${source.datatype} targetdt=${target.datatype}"
|
"source dt size must be less or equal to target dt size at $position srcdt=${source.datatype} targetdt=${target.datatype}"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@ -4,16 +4,25 @@ import prog8.code.core.*
|
|||||||
|
|
||||||
|
|
||||||
internal object DummyMemsizer : IMemSizer {
|
internal object DummyMemsizer : IMemSizer {
|
||||||
override fun memorySize(dt: DataType) = when(dt) {
|
override fun memorySize(dt: DataType, numElements: Int?): Int {
|
||||||
in ByteDatatypesWithBoolean -> 1
|
if(dt.isArray) {
|
||||||
DataType.FLOAT -> 5
|
require(numElements != null)
|
||||||
else -> 2
|
return when(dt.sub) {
|
||||||
|
BaseDataType.BOOL, BaseDataType.BYTE, BaseDataType.UBYTE -> numElements
|
||||||
|
BaseDataType.UWORD, BaseDataType.WORD -> numElements*2
|
||||||
|
BaseDataType.FLOAT -> numElements*5
|
||||||
|
else -> throw IllegalArgumentException("invalid sub type")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return when {
|
||||||
|
dt.isByteOrBool -> 1 * (numElements ?: 1)
|
||||||
|
dt.isFloat -> 5 * (numElements ?: 1)
|
||||||
|
else -> 2 * (numElements ?: 1)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
override fun memorySize(arrayDt: DataType, numElements: Int) = when(arrayDt) {
|
|
||||||
DataType.ARRAY_UW -> numElements*2
|
override fun memorySize(dt: BaseDataType): Int {
|
||||||
DataType.ARRAY_W -> numElements*2
|
return memorySize(DataType.forDt(dt), null)
|
||||||
DataType.ARRAY_F -> numElements*5
|
|
||||||
else -> numElements
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -8,6 +8,7 @@ import io.kotest.matchers.shouldBe
|
|||||||
import prog8.code.SymbolTableMaker
|
import prog8.code.SymbolTableMaker
|
||||||
import prog8.code.ast.*
|
import prog8.code.ast.*
|
||||||
import prog8.code.core.*
|
import prog8.code.core.*
|
||||||
|
import prog8.code.source.SourceCode
|
||||||
import prog8.code.target.C64Target
|
import prog8.code.target.C64Target
|
||||||
import prog8.codegen.cpu6502.AsmGen6502
|
import prog8.codegen.cpu6502.AsmGen6502
|
||||||
import java.nio.file.Files
|
import java.nio.file.Files
|
||||||
@ -26,7 +27,8 @@ class TestCodegen: FunSpec({
|
|||||||
floats = true,
|
floats = true,
|
||||||
noSysInit = false,
|
noSysInit = false,
|
||||||
compTarget = target,
|
compTarget = target,
|
||||||
loadAddress = target.machine.PROGRAM_LOAD_ADDRESS
|
loadAddress = target.machine.PROGRAM_LOAD_ADDRESS,
|
||||||
|
memtopAddress = 0xffffu
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -43,52 +45,84 @@ class TestCodegen: FunSpec({
|
|||||||
// xx += cx16.r0
|
// xx += cx16.r0
|
||||||
// }
|
// }
|
||||||
//}
|
//}
|
||||||
val codegen = AsmGen6502(prefixSymbols = false)
|
val codegen = AsmGen6502(prefixSymbols = false, 0)
|
||||||
val program = PtProgram("test", DummyMemsizer, DummyStringEncoder)
|
val program = PtProgram("test", DummyMemsizer, DummyStringEncoder)
|
||||||
val block = PtBlock("main",false, SourceCode.Generated("test"), PtBlock.Options(), Position.DUMMY)
|
val block = PtBlock("main",false, SourceCode.Generated("test"), PtBlock.Options(), Position.DUMMY)
|
||||||
val sub = PtSub("start", emptyList(), null, Position.DUMMY)
|
val sub = PtSub("start", emptyList(), null, Position.DUMMY)
|
||||||
sub.add(PtVariable("pi", DataType.UBYTE, ZeropageWish.DONTCARE, PtNumber(DataType.UBYTE, 0.0, Position.DUMMY), null, Position.DUMMY))
|
sub.add(PtVariable(
|
||||||
sub.add(PtVariable("particleX", DataType.ARRAY_UB, ZeropageWish.DONTCARE, null, 3u, Position.DUMMY))
|
"pi",
|
||||||
sub.add(PtVariable("particleDX", DataType.ARRAY_UB, ZeropageWish.DONTCARE, null, 3u, Position.DUMMY))
|
DataType.forDt(BaseDataType.UBYTE),
|
||||||
sub.add(PtVariable("xx", DataType.WORD, ZeropageWish.DONTCARE, PtNumber(DataType.WORD, 1.0, Position.DUMMY), null, Position.DUMMY))
|
ZeropageWish.DONTCARE,
|
||||||
|
0u,
|
||||||
|
PtNumber(BaseDataType.UBYTE, 0.0, Position.DUMMY),
|
||||||
|
null,
|
||||||
|
Position.DUMMY
|
||||||
|
))
|
||||||
|
sub.add(PtVariable(
|
||||||
|
"particleX",
|
||||||
|
DataType.arrayFor(BaseDataType.UBYTE),
|
||||||
|
ZeropageWish.DONTCARE,
|
||||||
|
0u,
|
||||||
|
null,
|
||||||
|
3u,
|
||||||
|
Position.DUMMY
|
||||||
|
))
|
||||||
|
sub.add(PtVariable(
|
||||||
|
"particleDX",
|
||||||
|
DataType.arrayFor(BaseDataType.UBYTE),
|
||||||
|
ZeropageWish.DONTCARE,
|
||||||
|
0u,
|
||||||
|
null,
|
||||||
|
3u,
|
||||||
|
Position.DUMMY
|
||||||
|
))
|
||||||
|
sub.add(PtVariable(
|
||||||
|
"xx",
|
||||||
|
DataType.forDt(BaseDataType.WORD),
|
||||||
|
ZeropageWish.DONTCARE,
|
||||||
|
0u,
|
||||||
|
PtNumber(BaseDataType.WORD, 1.0, Position.DUMMY),
|
||||||
|
null,
|
||||||
|
Position.DUMMY
|
||||||
|
))
|
||||||
|
|
||||||
val assign = PtAugmentedAssign("+=", Position.DUMMY)
|
val assign = PtAugmentedAssign("+=", Position.DUMMY)
|
||||||
val target = PtAssignTarget(Position.DUMMY).also {
|
val target = PtAssignTarget(false, Position.DUMMY).also {
|
||||||
val targetIdx = PtArrayIndexer(DataType.UBYTE, Position.DUMMY).also { idx ->
|
val targetIdx = PtArrayIndexer(DataType.forDt(BaseDataType.UBYTE), Position.DUMMY).also { idx ->
|
||||||
idx.add(PtIdentifier("main.start.particleX", DataType.ARRAY_UB, Position.DUMMY))
|
idx.add(PtIdentifier("main.start.particleX", DataType.arrayFor(BaseDataType.UBYTE), Position.DUMMY))
|
||||||
idx.add(PtNumber(DataType.UBYTE, 2.0, Position.DUMMY))
|
idx.add(PtNumber(BaseDataType.UBYTE, 2.0, Position.DUMMY))
|
||||||
}
|
}
|
||||||
it.add(targetIdx)
|
it.add(targetIdx)
|
||||||
}
|
}
|
||||||
val value = PtArrayIndexer(DataType.UBYTE, Position.DUMMY)
|
val value = PtArrayIndexer(DataType.forDt(BaseDataType.UBYTE), Position.DUMMY)
|
||||||
value.add(PtIdentifier("main.start.particleDX", DataType.ARRAY_UB, Position.DUMMY))
|
value.add(PtIdentifier("main.start.particleDX", DataType.arrayFor(BaseDataType.UBYTE), Position.DUMMY))
|
||||||
value.add(PtNumber(DataType.UBYTE, 2.0, Position.DUMMY))
|
value.add(PtNumber(BaseDataType.UBYTE, 2.0, Position.DUMMY))
|
||||||
assign.add(target)
|
assign.add(target)
|
||||||
assign.add(value)
|
assign.add(value)
|
||||||
sub.add(assign)
|
sub.add(assign)
|
||||||
|
|
||||||
val prefixAssign = PtAugmentedAssign("-", Position.DUMMY)
|
val prefixAssign = PtAugmentedAssign("-", Position.DUMMY)
|
||||||
val prefixTarget = PtAssignTarget(Position.DUMMY).also {
|
val prefixTarget = PtAssignTarget(false, Position.DUMMY).also {
|
||||||
it.add(PtIdentifier("main.start.xx", DataType.WORD, Position.DUMMY))
|
it.add(PtIdentifier("main.start.xx", DataType.forDt(BaseDataType.WORD), Position.DUMMY))
|
||||||
}
|
}
|
||||||
prefixAssign.add(prefixTarget)
|
prefixAssign.add(prefixTarget)
|
||||||
prefixAssign.add(PtIdentifier("main.start.xx", DataType.WORD, Position.DUMMY))
|
prefixAssign.add(PtIdentifier("main.start.xx", DataType.forDt(BaseDataType.WORD), Position.DUMMY))
|
||||||
sub.add(prefixAssign)
|
sub.add(prefixAssign)
|
||||||
|
|
||||||
val numberAssign = PtAugmentedAssign("-=", Position.DUMMY)
|
val numberAssign = PtAugmentedAssign("-=", Position.DUMMY)
|
||||||
val numberAssignTarget = PtAssignTarget(Position.DUMMY).also {
|
val numberAssignTarget = PtAssignTarget(false, Position.DUMMY).also {
|
||||||
it.add(PtIdentifier("main.start.xx", DataType.WORD, Position.DUMMY))
|
it.add(PtIdentifier("main.start.xx", DataType.forDt(BaseDataType.WORD), Position.DUMMY))
|
||||||
}
|
}
|
||||||
numberAssign.add(numberAssignTarget)
|
numberAssign.add(numberAssignTarget)
|
||||||
numberAssign.add(PtNumber(DataType.WORD, 42.0, Position.DUMMY))
|
numberAssign.add(PtNumber(BaseDataType.WORD, 42.0, Position.DUMMY))
|
||||||
sub.add(numberAssign)
|
sub.add(numberAssign)
|
||||||
|
|
||||||
val cxregAssign = PtAugmentedAssign("+=", Position.DUMMY)
|
val cxregAssign = PtAugmentedAssign("+=", Position.DUMMY)
|
||||||
val cxregAssignTarget = PtAssignTarget(Position.DUMMY).also {
|
val cxregAssignTarget = PtAssignTarget(false, Position.DUMMY).also {
|
||||||
it.add(PtIdentifier("main.start.xx", DataType.WORD, Position.DUMMY))
|
it.add(PtIdentifier("main.start.xx", DataType.forDt(BaseDataType.WORD), Position.DUMMY))
|
||||||
}
|
}
|
||||||
cxregAssign.add(cxregAssignTarget)
|
cxregAssign.add(cxregAssignTarget)
|
||||||
cxregAssign.add(PtIdentifier("cx16.r0", DataType.UWORD, Position.DUMMY))
|
cxregAssign.add(PtIdentifier("cx16.r0", DataType.forDt(BaseDataType.UWORD), Position.DUMMY))
|
||||||
sub.add(cxregAssign)
|
sub.add(cxregAssign)
|
||||||
|
|
||||||
block.add(sub)
|
block.add(sub)
|
||||||
@ -96,7 +130,7 @@ class TestCodegen: FunSpec({
|
|||||||
|
|
||||||
// define the "cx16.r0" virtual register
|
// define the "cx16.r0" virtual register
|
||||||
val cx16block = PtBlock("cx16", false, SourceCode.Generated("test"), PtBlock.Options(), Position.DUMMY)
|
val cx16block = PtBlock("cx16", false, SourceCode.Generated("test"), PtBlock.Options(), Position.DUMMY)
|
||||||
cx16block.add(PtMemMapped("r0", DataType.UWORD, 100u, null, Position.DUMMY))
|
cx16block.add(PtMemMapped("r0", DataType.forDt(BaseDataType.UWORD), 100u, null, Position.DUMMY))
|
||||||
program.add(cx16block)
|
program.add(cx16block)
|
||||||
|
|
||||||
val options = getTestOptions()
|
val options = getTestOptions()
|
||||||
|
@ -1,47 +0,0 @@
|
|||||||
|
|
||||||
plugins {
|
|
||||||
id 'java'
|
|
||||||
id 'application'
|
|
||||||
id "org.jetbrains.kotlin.jvm"
|
|
||||||
}
|
|
||||||
|
|
||||||
java {
|
|
||||||
toolchain {
|
|
||||||
languageVersion = JavaLanguageVersion.of(javaVersion)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
compileKotlin {
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = javaVersion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
compileTestKotlin {
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = javaVersion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
dependencies {
|
|
||||||
implementation project(':codeCore')
|
|
||||||
implementation project(':intermediate')
|
|
||||||
implementation project(':codeGenIntermediate')
|
|
||||||
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk8"
|
|
||||||
// implementation "org.jetbrains.kotlin:kotlin-reflect"
|
|
||||||
implementation "com.michael-bull.kotlin-result:kotlin-result-jvm:1.1.20"
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
sourceSets {
|
|
||||||
main {
|
|
||||||
java {
|
|
||||||
srcDir "${project.projectDir}/src"
|
|
||||||
}
|
|
||||||
resources {
|
|
||||||
srcDir "${project.projectDir}/res"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// note: there are no unit tests in this module!
|
|
27
codeGenExperimental/build.gradle.kts
Normal file
27
codeGenExperimental/build.gradle.kts
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
import org.jetbrains.kotlin.gradle.dsl.JvmTarget
|
||||||
|
|
||||||
|
plugins {
|
||||||
|
kotlin("jvm")
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
implementation(project(":codeCore"))
|
||||||
|
implementation(project(":intermediate"))
|
||||||
|
implementation(project(":codeGenIntermediate"))
|
||||||
|
// implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk8")
|
||||||
|
// implementation "org.jetbrains.kotlin:kotlin-reflect"
|
||||||
|
implementation("com.michael-bull.kotlin-result:kotlin-result-jvm:2.0.0")
|
||||||
|
}
|
||||||
|
|
||||||
|
sourceSets {
|
||||||
|
main {
|
||||||
|
java {
|
||||||
|
srcDir(file("${project.projectDir}/src"))
|
||||||
|
}
|
||||||
|
resources {
|
||||||
|
srcDir(file("${project.projectDir}/res"))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// note: there are no unit tests in this module!
|
@ -1,66 +0,0 @@
|
|||||||
|
|
||||||
plugins {
|
|
||||||
id 'java'
|
|
||||||
id 'application'
|
|
||||||
id "org.jetbrains.kotlin.jvm"
|
|
||||||
}
|
|
||||||
|
|
||||||
java {
|
|
||||||
toolchain {
|
|
||||||
languageVersion = JavaLanguageVersion.of(javaVersion)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
compileKotlin {
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = javaVersion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
compileTestKotlin {
|
|
||||||
kotlinOptions {
|
|
||||||
jvmTarget = javaVersion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
dependencies {
|
|
||||||
implementation project(':codeCore')
|
|
||||||
implementation project(':intermediate')
|
|
||||||
implementation "org.jetbrains.kotlin:kotlin-stdlib-jdk8"
|
|
||||||
// implementation "org.jetbrains.kotlin:kotlin-reflect"
|
|
||||||
implementation "com.michael-bull.kotlin-result:kotlin-result-jvm:1.1.20"
|
|
||||||
|
|
||||||
testImplementation 'io.kotest:kotest-runner-junit5-jvm:5.8.0'
|
|
||||||
testImplementation 'org.junit.jupiter:junit-jupiter:5.9.2'
|
|
||||||
testRuntimeOnly 'org.junit.platform:junit-platform-launcher'
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
sourceSets {
|
|
||||||
main {
|
|
||||||
java {
|
|
||||||
srcDir "${project.projectDir}/src"
|
|
||||||
}
|
|
||||||
resources {
|
|
||||||
srcDir "${project.projectDir}/res"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
test {
|
|
||||||
java {
|
|
||||||
srcDir "${project.projectDir}/test"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
test {
|
|
||||||
// Enable JUnit 5 (Gradle 4.6+).
|
|
||||||
useJUnitPlatform()
|
|
||||||
|
|
||||||
// Always run tests, even when nothing changed.
|
|
||||||
dependsOn 'cleanTest'
|
|
||||||
|
|
||||||
// Show test results.
|
|
||||||
testLogging {
|
|
||||||
events "skipped", "failed"
|
|
||||||
}
|
|
||||||
}
|
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user