mirror of
				https://github.com/irmen/prog8.git
				synced 2025-10-25 05:18:38 +00:00 
			
		
		
		
	Compare commits
	
		
			3041 Commits
		
	
	
		
			v7.5
			...
			v12.0-beta
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 231b50dacb | ||
|  | a71895cbe8 | ||
|  | a8bede17b2 | ||
|  | f6c8e693a5 | ||
|  | 9461e4088c | ||
|  | efd73fd10d | ||
|  | ddf6e84a1a | ||
|  | 633d6c34e2 | ||
|  | 6e7fbc6683 | ||
|  | 124ea1230b | ||
|  | 8b48a295b6 | ||
|  | d285d37fdb | ||
|  | 8bb927b483 | ||
|  | 1af4cd0d63 | ||
|  | db2f28c4cd | ||
|  | 5d9fbd2ccc | ||
|  | 7efc709538 | ||
|  | 79419a98d0 | ||
|  | 1c77d5d5e7 | ||
|  | c6854e22a3 | ||
|  | 83acc2f285 | ||
|  | 6de95f7a3b | ||
|  | 82839a2d82 | ||
|  | e5fc9b3132 | ||
|  | ced4c5944a | ||
|  | d4c460072b | ||
|  | 0b9384b556 | ||
|  | 6c3277e3e3 | ||
|  | d9ff1eb38a | ||
|  | e178097735 | ||
|  | a1ab8ed208 | ||
|  | 6ababbf8f4 | ||
|  | 79629befc1 | ||
|  | 8022c0772a | ||
|  | 8ad2b4638b | ||
|  | 3a0392df8a | ||
|  | beb28b061d | ||
|  | c39acc5031 | ||
|  | f54a29c415 | ||
|  | 783b111059 | ||
|  | fb286f8b54 | ||
|  | 5999110e3f | ||
|  | 52a757ea78 | ||
|  | 28df08eea8 | ||
|  | 79505308ba | ||
|  | a7e9d8e14b | ||
|  | 08f3abe5bf | ||
|  | 3ef09d7d9a | ||
|  | a9142b9ce5 | ||
|  | 13e6f64d3b | ||
|  | 4a1256c772 | ||
|  | ff9bec90ec | ||
|  | a6fee1e510 | ||
|  | 80538f101e | ||
|  | aee53b14c7 | ||
|  | 5eb2fc8d86 | ||
|  | 98f91bbf88 | ||
|  | b48b36ef18 | ||
|  | 221a093e5f | ||
|  | 2ca1820d4e | ||
|  | d58737d5be | ||
|  | b52cee3154 | ||
|  | 9a76941e10 | ||
|  | 0285a4cce1 | ||
|  | 5a3aa1bd25 | ||
|  | 0f79351de9 | ||
|  | 3cdb25ce8e | ||
|  | b7193bd0c6 | ||
|  | 10ff6a0095 | ||
|  | d30f58004e | ||
|  | 17bdb22e4f | ||
|  | a68209be37 | ||
|  | 7b40eade44 | ||
|  | 8a717c74b9 | ||
|  | aa72ded21e | ||
|  | 8e53c83844 | ||
|  | e2a2db1256 | ||
|  | 2afd2d4dae | ||
|  | fad17cc094 | ||
|  | f93d957999 | ||
|  | 8db0344cee | ||
|  | 32e531f951 | ||
|  | a4769702f9 | ||
|  | cf19fb8df1 | ||
|  | 79b8bb5c9f | ||
|  | fc5889ec0b | ||
|  | 369303c46d | ||
|  | d65670cc7b | ||
|  | f74eeaee0f | ||
|  | 826fb3e9c2 | ||
|  | a3d7b8a899 | ||
|  | 0cc36ed6e4 | ||
|  | 976bd52972 | ||
|  | 4a8d5def84 | ||
|  | 2f60716082 | ||
|  | 729efb04e1 | ||
|  | 4ea8b4d445 | ||
|  | e800c165f9 | ||
|  | fd9bd23449 | ||
|  | 8880ed1393 | ||
|  | f7fde070ca | ||
|  | 5ada80779d | ||
|  | 8972235a0e | ||
|  | e56f533e38 | ||
|  | 324fb7dbf7 | ||
|  | 44285b9b5d | ||
|  | a68f477d61 | ||
|  | ae9f99448e | ||
|  | 7c0fb10197 | ||
|  | 9e85571a7b | ||
|  | 9e10c15e2e | ||
|  | 6bd7752bac | ||
|  | 83ec437e8a | ||
|  | 4a1d05dd46 | ||
|  | aa324e355a | ||
|  | 5cb8bcead7 | ||
|  | bbd06c0c99 | ||
|  | 651830ea82 | ||
|  | c70146f1dc | ||
|  | d4e83b28bb | ||
|  | bc58a25765 | ||
|  | 38645022c9 | ||
|  | 647cd0fbe1 | ||
|  | ea8935a346 | ||
|  | 7ea80babfc | ||
|  | dee761a99e | ||
|  | 88ee7a8187 | ||
|  | eb8b408b82 | ||
|  | 3d10882f57 | ||
|  | 1988496512 | ||
|  | 88b074c208 | ||
|  | c4c5636a81 | ||
|  | c39d570b72 | ||
|  | 4ccd7f9f3a | ||
|  | 1c9c5aeef7 | ||
|  | 23ad540aa5 | ||
|  | 08810c2749 | ||
|  | a52966f327 | ||
|  | 624220e9a3 | ||
|  | 842b11ed9e | ||
|  | 82267b3f56 | ||
|  | 67fb45a55b | ||
|  | 11186f1dbe | ||
|  | 0116fac201 | ||
|  | 817f4f8e7c | ||
|  | 866313209b | ||
|  | 28e351daab | ||
|  | 893e16d814 | ||
|  | 33470c47fc | ||
|  | 63f7b87572 | ||
|  | e2901cca1b | ||
|  | ce8006992a | ||
|  | 0b5413ad83 | ||
|  | dd7adde387 | ||
|  | 23058b51a1 | ||
|  | 2f90c53ad0 | ||
|  | c3be7ab4b3 | ||
|  | a9b8fbc6c6 | ||
|  | 720988ae72 | ||
|  | b0981a5fae | ||
|  | ea5deeefbd | ||
|  | 054c98da7c | ||
|  | dc434d034a | ||
|  | a4a1b563aa | ||
|  | 3f34b83e0d | ||
|  | 9c63ef39c7 | ||
|  | 9f6106452e | ||
|  | f9fbfe30e3 | ||
|  | 9a9bf170c6 | ||
|  | 7dd64b4f13 | ||
|  | b6c0bac96f | ||
|  | 8ede098154 | ||
|  | 2a4a3b786e | ||
|  | b4e0a2019e | ||
|  | e14c3f8b59 | ||
|  | c81f76226d | ||
|  | edc353cc24 | ||
|  | dcce519c69 | ||
|  | 0a16dcafc0 | ||
|  | 54d41b7f6f | ||
|  | 0541b84d09 | ||
|  | 1b420f7fe7 | ||
|  | 6a9a82ff9d | ||
|  | aa36e6b19f | ||
|  | 51cb6aad50 | ||
|  | b5ce409592 | ||
|  | 2119817e4a | ||
|  | 1efdfe8ea1 | ||
|  | 67d4180825 | ||
|  | be31e190d2 | ||
|  | a68cf3c812 | ||
|  | c2bf9024f8 | ||
|  | bd72eaad4c | ||
|  | b5d1575823 | ||
|  | 1d306e5cdc | ||
|  | b137164fe6 | ||
|  | 67d4ad50e1 | ||
|  | c71066af4c | ||
|  | 6f0a0981bd | ||
|  | 49a4d9ba37 | ||
|  | fcdfa741b9 | ||
|  | e3e395836d | ||
|  | 3bab177d50 | ||
|  | 12abafb917 | ||
|  | 8dc2e47507 | ||
|  | 0be90dedf2 | ||
|  | daf7c3357c | ||
|  | e6bab3ceeb | ||
|  | 59387b2ae8 | ||
|  | e8795859c5 | ||
|  | bebe60b687 | ||
|  | ddceec364e | ||
|  | f8f20440d3 | ||
|  | f8722faa4e | ||
|  | d067fa4b73 | ||
|  | 26fbbf48a4 | ||
|  | d5cc414221 | ||
|  | b5e51ab937 | ||
|  | 552e55c29f | ||
|  | a228908c1a | ||
|  | 15fc3b6c04 | ||
|  | 0456badd02 | ||
|  | d28f154f1c | ||
|  | 399cf5118d | ||
|  | a87f2640d3 | ||
|  | a90ef274d7 | ||
|  | 1c02179c5c | ||
|  | 77584493fd | ||
|  | a36709e638 | ||
|  | 341778ba67 | ||
|  | 8d63cce749 | ||
|  | ec50b5a007 | ||
|  | 8e7bbcdbe0 | ||
|  | 37ecdc47b3 | ||
|  | 112ca3cc53 | ||
|  | 33b3a1664c | ||
|  | 8a0c02e264 | ||
|  | 31d84c8921 | ||
|  | 34bedbeef1 | ||
|  | 3b1b0985c1 | ||
|  | e40ace9dea | ||
|  | 4c0e6e2640 | ||
|  | 08b314c37d | ||
|  | 86da9d3c7e | ||
|  | 4e61e25c02 | ||
|  | 5097d52d99 | ||
|  | 368387e1a7 | ||
|  | 09d2185bb1 | ||
|  | 5c02e2bd71 | ||
|  | fb01389b3d | ||
|  | aaa81210ce | ||
|  | 51269257ea | ||
|  | 23a853db1e | ||
|  | 9da430ffeb | ||
|  | cc063124cf | ||
|  | 3b37b89951 | ||
|  | 844b537d1e | ||
|  | caf1d4a22a | ||
|  | d8e244df99 | ||
|  | 548e421e27 | ||
|  | 322fa7ea69 | ||
|  | db6c887795 | ||
|  | cf7bea0985 | ||
|  | 61fe55168a | ||
|  | 25d7f8808f | ||
|  | 1c4999ec87 | ||
|  | c726d3f937 | ||
|  | f70341df1b | ||
|  | f0b791452e | ||
|  | adf5600a9b | ||
|  | 6d4ccc5feb | ||
|  | 5f3829d5cc | ||
|  | 770ebdcd4a | ||
|  | 96f690e749 | ||
|  | eabdd3a8f3 | ||
|  | 50650b966b | ||
|  | 65e34d4989 | ||
|  | 05dad5ab5f | ||
|  | 1a69a2f1bc | ||
|  | 435faafaad | ||
|  | 686b32dc29 | ||
|  | 0e64a22910 | ||
|  | 4f0839f27e | ||
|  | bb1953267d | ||
|  | acc630972a | ||
|  | 6a33be3fd8 | ||
|  | cd8aae4681 | ||
|  | 11456496bd | ||
|  | f5fc4e345c | ||
|  | 86eef7039f | ||
|  | f4b2264fcf | ||
|  | 9b36ae2277 | ||
|  | 913ab03963 | ||
|  | 38448e471c | ||
|  | 67231af623 | ||
|  | e31ef6f06f | ||
|  | 09d188106a | ||
|  | d8e2116481 | ||
|  | 435dfbb932 | ||
|  | ba93966474 | ||
|  | ea8d17cdb2 | ||
|  | 082265fb25 | ||
|  | d138a7a567 | ||
|  | ea27d732ab | ||
|  | 9e557ce8ac | ||
|  | 924e28e9b3 | ||
|  | e5d9af75de | ||
|  | 31c1bf8bc5 | ||
|  | 37d4055036 | ||
|  | 78b1076110 | ||
|  | 0a3c748e41 | ||
|  | ebf79ef9e2 | ||
|  | 60a73248cd | ||
|  | abbb7d7ba3 | ||
|  | 59c378089e | ||
|  | 0b789b5f0b | ||
|  | 4382b96a9a | ||
|  | 246e4f35a6 | ||
|  | 99b9370178 | ||
|  | 506062c6b6 | ||
|  | d634061cd9 | ||
|  | 8353c689ca | ||
|  | d59d8ff1fe | ||
|  | e98e6f70ac | ||
|  | 53e442d509 | ||
|  | 134352ed7c | ||
|  | f7cbfdff06 | ||
|  | b28ee0819f | ||
|  | 5de626aab8 | ||
|  | 7aad5d486e | ||
|  | 701f155951 | ||
|  | 8c324d7514 | ||
|  | 522958e0e9 | ||
|  | 97390db5f5 | ||
|  | af920d1427 | ||
|  | 779ebc0537 | ||
|  | 38949b82c3 | ||
|  | d11386ef26 | ||
|  | 0e0377d1f0 | ||
|  | 55e0dbab27 | ||
|  | 4dc82f2c83 | ||
|  | 1ba5587404 | ||
|  | 835c4b6da3 | ||
|  | dbd955b61e | ||
|  | d20e2fd88c | ||
|  | e0dea89477 | ||
|  | ccc6b56e35 | ||
|  | 6fc2902895 | ||
|  | c96e4b40d4 | ||
|  | 37da3e2170 | ||
|  | 2661d3c489 | ||
|  | b89bbb9281 | ||
|  | 696bf636ed | ||
|  | 40952a788a | ||
|  | 0162e7a0c1 | ||
|  | 6ce099f176 | ||
|  | 476a4bac8e | ||
|  | 63a410a6df | ||
|  | cca27faa3b | ||
|  | 803e6bd81a | ||
|  | 88269628a2 | ||
|  | b920d553a0 | ||
|  | 5e2d0d0dfc | ||
|  | 2ae3bd68eb | ||
|  | 9c183f27eb | ||
|  | 8046023e82 | ||
|  | e328520588 | ||
|  | 7eb079050c | ||
|  | 2fdd5543b2 | ||
|  | d04164c0a6 | ||
|  | b047731f82 | ||
|  | 4d91f92a2e | ||
|  | 98505d27b1 | ||
|  | cd63a58ad9 | ||
|  | 170f8dd092 | ||
|  | 619dcb6a84 | ||
|  | 99ae8ea52e | ||
|  | dc031c30eb | ||
|  | 1e702439b7 | ||
|  | 8debc42381 | ||
|  | 532d719089 | ||
|  | b40860aca4 | ||
|  | 2cbe6b5f7f | ||
|  | d2cc7ccdfa | ||
|  | 2cb183c6d8 | ||
|  | 84026b105f | ||
|  | a4d0589f10 | ||
|  | e375f6afce | ||
|  | 5a7bc04816 | ||
|  | bd1894580e | ||
|  | 9e694c0337 | ||
|  | c82586db28 | ||
|  | dd2d466350 | ||
|  | 830da8de0a | ||
|  | 4e5ee333c8 | ||
|  | 9df899eb63 | ||
|  | ca7491a702 | ||
|  | 1a07129865 | ||
|  | 4fbd67ff99 | ||
|  | 5bc6c50f42 | ||
|  | 063de3801d | ||
|  | ae65266a4a | ||
|  | 8ed2401e0b | ||
|  | d2e8ee8269 | ||
|  | 1f996e3b8b | ||
|  | 7108b74105 | ||
|  | 801fe1b604 | ||
|  | fb44c87597 | ||
|  | 6b9cdbd482 | ||
|  | 0ab98033b5 | ||
|  | 14a2b96609 | ||
|  | f829b689db | ||
|  | dfda8b7ed5 | ||
|  | 4388466451 | ||
|  | 5c2f509a52 | ||
|  | 59582f5210 | ||
|  | e2a8bdbdfb | ||
|  | 0916b943da | ||
|  | 9c7ebc883c | ||
|  | 0ee42b9aa0 | ||
|  | 37b3868ca3 | ||
|  | a6835ce3f0 | ||
|  | 69c96ad99b | ||
|  | b72877d59d | ||
|  | 05eb15d4f7 | ||
|  | f1fec37c79 | ||
|  | 73f6880ff8 | ||
|  | 8a53742f31 | ||
|  | 9be40e85ff | ||
|  | 61079c1eb7 | ||
|  | 1075ee8fc3 | ||
|  | a28b265197 | ||
|  | 20e534c468 | ||
|  | da7aa5dc49 | ||
|  | 8f2a43ca0a | ||
|  | d0909d7810 | ||
|  | 1641999d20 | ||
|  | e16452037c | ||
|  | 344d79684a | ||
|  | 573a1d9b7b | ||
|  | 25ab57580c | ||
|  | a332e0e3d1 | ||
|  | 376f1cb139 | ||
|  | 90f80558d7 | ||
|  | e281994898 | ||
|  | 29fac122e1 | ||
|  | 1dc412eb90 | ||
|  | 3770a4fe0c | ||
|  | 79cda544c8 | ||
|  | f04b97d890 | ||
|  | 3e9b4ccc45 | ||
|  | 2c3d838dd8 | ||
|  | 7668a3c660 | ||
|  | 5dd45b714a | ||
|  | 8b08895d0f | ||
|  | 8f8d99e3ed | ||
|  | 23474360ec | ||
|  | 81c255c450 | ||
|  | ef23d52ed7 | ||
|  | 220ab773aa | ||
|  | e3e5bff7bb | ||
|  | 7b9a841b2a | ||
|  | 40423911ef | ||
|  | 582a70b046 | ||
|  | 5b63590ebf | ||
|  | 2b6510dc19 | ||
|  | 9d49589d73 | ||
|  | 125b66c929 | ||
|  | 5255f1c052 | ||
|  | a6ba05d60c | ||
|  | 41e963b04b | ||
|  | 6ff75bef29 | ||
|  | 72c16d0d32 | ||
|  | 94653e5c8c | ||
|  | 3e2b2a698d | ||
|  | ae04f5aee8 | ||
|  | 5c56267662 | ||
|  | e55ce5504e | ||
|  | fb1e89d9ef | ||
|  | bc550a4549 | ||
|  | ebdea9cf76 | ||
|  | 09ec508f82 | ||
|  | d06e9ea7f6 | ||
|  | a36bdc54fd | ||
|  | 0814ea9711 | ||
|  | daefe839d8 | ||
|  | e6088dd315 | ||
|  | fc03d6f332 | ||
|  | 2aeb7a838e | ||
|  | 99ff5dd078 | ||
|  | 49982b49b6 | ||
|  | fd39c22616 | ||
|  | 9e79722a7f | ||
|  | 17334a1c58 | ||
|  | c7f0ff11ac | ||
|  | cd2cc89e6a | ||
|  | 069143092d | ||
|  | efd41260f2 | ||
|  | 8d2410622c | ||
|  | 60554389b3 | ||
|  | a940dc7d43 | ||
|  | 06ca68a625 | ||
|  | 5b58e5b158 | ||
|  | 74dd8fe80b | ||
|  | 75ddcda5f3 | ||
|  | 216825b98a | ||
|  | a96defab86 | ||
|  | 0864b0a1b7 | ||
|  | 8b158d9240 | ||
|  | f335251c2b | ||
|  | 67bc0b6931 | ||
|  | e646dd1ed1 | ||
|  | 2b7947f9b0 | ||
|  | ec0cfb4b3f | ||
|  | 9cdf53019c | ||
|  | 1a04a3eb3a | ||
|  | 105d3995e0 | ||
|  | 8ce3204f93 | ||
|  | d0f15f1285 | ||
|  | 66d6f67120 | ||
|  | a106c88054 | ||
|  | ee784e1ccc | ||
|  | bb75be0b44 | ||
|  | 2478aea316 | ||
|  | 1e17df5296 | ||
|  | 8583a96519 | ||
|  | d0c184c7de | ||
|  | 0191acb2b3 | ||
|  | 277a1a32b2 | ||
|  | 7a13f57ab0 | ||
|  | 0c882836d9 | ||
|  | 228be5cd04 | ||
|  | 08cd2fd6e8 | ||
|  | bc7b086f0f | ||
|  | e8f3af6981 | ||
|  | f9c7c7dab7 | ||
|  | 09a17743ad | ||
|  | 4f096a7511 | ||
|  | 2ab2130000 | ||
|  | 66558f7638 | ||
|  | a6f9ed07e7 | ||
|  | 7268a8736f | ||
|  | 8f6b5676d7 | ||
|  | ca9422bbe9 | ||
|  | 35d9412559 | ||
|  | f071c07dd9 | ||
|  | e5ff3c1ff3 | ||
|  | f0e8ff0326 | ||
|  | 3b5cda85ff | ||
|  | 420793f9e2 | ||
|  | cf1dbaf0d8 | ||
|  | d187cef6b7 | ||
|  | 3b3616afda | ||
|  | 0ffebc25d0 | ||
|  | 478e2b4ebd | ||
|  | a56ae7539a | ||
|  | 407773bda2 | ||
|  | 823eaa8918 | ||
|  | a2be42c5ca | ||
|  | a76b8d66ff | ||
|  | b7f47d354f | ||
|  | 5d33c93af9 | ||
|  | 4db6859f3f | ||
|  | 45fe1bb16e | ||
|  | b014facbd3 | ||
|  | 3b4b37f16b | ||
|  | 68d5983a14 | ||
|  | f2cfcfdf31 | ||
|  | 10b9162dc5 | ||
|  | c84cc8f8c9 | ||
|  | 78c71bbf0e | ||
|  | 37c2c1bf0b | ||
|  | c8996418da | ||
|  | 76b29aa629 | ||
|  | ee521793f8 | ||
|  | f42e12bc13 | ||
|  | 427451a23f | ||
|  | af7930d494 | ||
|  | e2882d37bf | ||
|  | 942d3ee640 | ||
|  | 7b4a82b91a | ||
|  | 056c0a24d9 | ||
|  | 827df04b32 | ||
|  | e174b31344 | ||
|  | 49959af752 | ||
|  | 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 | ||
|  | fe2b67998c | ||
|  | 04df3c9f7f | ||
|  | de3d0b40dc | ||
|  | 4db4a5f1b2 | ||
|  | 5a0524ff4d | ||
|  | 5b7801eea1 | ||
|  | fbe231793b | ||
|  | 6a9269111e | ||
|  | a94cfd34f5 | ||
|  | 28eae5a0fd | ||
|  | 1818738fc8 | ||
|  | 7e1e7a0780 | ||
|  | 1fc79ff6dd | ||
|  | 3535c1acda | ||
|  | 33c8caac8f | ||
|  | 51d708bbdd | ||
|  | a5a918df84 | ||
|  | 820541e427 | ||
|  | e63a8f0c01 | ||
|  | c11a9b8709 | ||
|  | 80f39e8097 | ||
|  | 2a8b65e29c | ||
|  | 4bdf50145e | ||
|  | 3a9919a377 | ||
|  | eef8ae00b8 | ||
|  | ed15fac691 | ||
|  | f739e679e4 | ||
|  | fc0fae8caf | ||
|  | f46896fd74 | ||
|  | 52649a8e4f | ||
|  | bdfb01f6a0 | ||
|  | 1137e57393 | ||
|  | 267ea13e8c | ||
|  | 04f7b772a3 | ||
|  | 42c7569791 | ||
|  | 6d29b00a80 | ||
|  | 9f1bd2d7d6 | ||
|  | 9826d7c494 | ||
|  | c6bf57b390 | ||
|  | bfcf07c1a2 | ||
|  | 4d7e96d423 | ||
|  | 449461e412 | ||
|  | 607275ec66 | ||
|  | e55cde2a81 | ||
|  | 84afb374e6 | ||
|  | da1620807f | ||
|  | f39ef8f565 | ||
|  | fe8b6e820c | ||
|  | f29d24e96a | ||
|  | 620ffe54ec | ||
|  | ceaa4cd07d | ||
|  | af17f903ee | ||
|  | c532e28841 | ||
|  | dba0846866 | ||
|  | bed629998a | ||
|  | bc2ede76bf | ||
|  | 2a1fec2ed2 | ||
|  | 004048e5a7 | ||
|  | b941d6f1e4 | ||
|  | 37b346740b | ||
|  | f5e332daf7 | ||
|  | fe9a9fc5cb | ||
|  | cc57477b99 | ||
|  | a1574a7187 | ||
|  | a5110b1f96 | ||
|  | 006713fe13 | ||
|  | 7868e672e0 | ||
|  | e1a133c2c0 | ||
|  | c77cd0da39 | ||
|  | 577333f2c4 | ||
|  | 7d8cdcbfea | ||
|  | c5c4c6f111 | ||
|  | 73be754680 | ||
|  | acd841dbb6 | ||
|  | 6b52ba9397 | ||
|  | 10d12f73d6 | ||
|  | cd9119655c | ||
|  | 41afeccd51 | ||
|  | 6b87cbb703 | ||
|  | 32afcbfe42 | ||
|  | bc2b38daf4 | ||
|  | f40b7b62bb | ||
|  | 1ca3f64bf0 | ||
|  | 92527b4c1d | ||
|  | c48012c385 | ||
|  | a282b17286 | ||
|  | 58d9463f16 | ||
|  | 047decd552 | ||
|  | 82e0877e64 | ||
|  | 040d75dafa | ||
|  | 4e1686f6e3 | ||
|  | b5e691f367 | ||
|  | 325f55f22d | ||
|  | 9724f2db7d | ||
|  | 5f20f321f0 | ||
|  | d4b087ea3f | ||
|  | 8ff10724d1 | ||
|  | 1581381467 | ||
|  | 96b5a30f60 | ||
|  | 0e17a0474a | ||
|  | b27368175d | ||
|  | aba36f7c92 | ||
|  | a3fa946300 | ||
|  | 01bbc2234e | ||
|  | 58e1864144 | ||
|  | 88458f5355 | ||
|  | a4f697bae1 | ||
|  | 8201408f16 | ||
|  | 8b8caa1c2e | ||
|  | 4dc50cb551 | ||
|  | 5522a305ab | ||
|  | d7f72056fc | ||
|  | 64c9c9b7fe | ||
|  | 98e1c843e4 | ||
|  | 906d9d858c | ||
|  | 16c1309df1 | ||
|  | 6eacf1bddd | ||
|  | 6c8c8e11cc | ||
|  | e941d2665a | ||
|  | 68669dbef0 | ||
|  | 6a48de9a9f | ||
|  | 9d6d98930b | ||
|  | 3cc858db12 | ||
|  | 386a391fd9 | ||
|  | d33aed4ed5 | ||
|  | 73ec8c31ad | ||
|  | 24944ad49e | ||
|  | 26ed231f61 | ||
|  | 8485b8429f | ||
|  | 358215e4dd | ||
|  | f874942075 | ||
|  | 2cadb546d5 | ||
|  | 344a1b9eb8 | ||
|  | 3c77f8a020 | ||
|  | 8e00408e3e | ||
|  | abcdfd8e28 | ||
|  | b0f5b6925d | ||
|  | ef79d0c43e | ||
|  | 78b4288005 | ||
|  | 680f5d21ee | ||
|  | c71aa0895f | ||
|  | 9f8e61789a | ||
|  | 932035cdc5 | ||
|  | ef198f1493 | ||
|  | 48ef856c0b | ||
|  | 9aea2b22c4 | ||
|  | e0055bc431 | ||
|  | 9553248ed6 | ||
|  | 39d2194d8f | ||
|  | 0800033b47 | ||
|  | 64d8943b7d | ||
|  | 444e97b00b | ||
|  | 1816bda7ea | ||
|  | d4a2031c07 | ||
|  | 8cf0b6cf51 | ||
|  | f2010bf7a5 | ||
|  | 8f56a7fe69 | ||
|  | 64c132ee0a | ||
|  | 84a7e86fe3 | ||
|  | a8c09d6144 | ||
|  | 87c46ba730 | ||
|  | 0f83dc6491 | ||
|  | cc22861719 | ||
|  | a14c192ea3 | ||
|  | b3d98be862 | ||
|  | 43027a4728 | ||
|  | 03831a7394 | ||
|  | fdbbd181ea | ||
|  | 69075376dc | ||
|  | 504d1440cc | ||
|  | 9e33b8b8da | ||
|  | 0cfcc5cd29 | ||
|  | e0de662f8e | ||
|  | 66a836d094 | ||
|  | 80095f4962 | ||
|  | 828d83dbef | ||
|  | 7de665d1e4 | ||
|  | 0a356ba73a | ||
|  | 41de8caa13 | ||
|  | 968609d06d | ||
|  | 3b199a2a87 | ||
|  | 0c1018ec61 | ||
|  | bc3f2db3de | ||
|  | 06bedb7adb | ||
|  | 45a9751217 | ||
|  | e8da62aa29 | ||
|  | ddb2ff4216 | ||
|  | f27e3478b9 | ||
|  | 38dc7fb7bd | ||
|  | aa4cd13c31 | ||
|  | 4a4b6c04a1 | ||
|  | 37fa3b34a2 | ||
|  | f8084e7955 | ||
|  | 4d5119ce3e | ||
|  | d85c347a6c | ||
|  | 7dd758a753 | ||
|  | 806654fc44 | ||
|  | 8e6b91cb9e | ||
|  | 334e6dca28 | ||
|  | f2daa17b92 | ||
|  | 6d9fccacb1 | ||
|  | 37638e7ed0 | ||
|  | 8a0e650511 | ||
|  | 8ba5a0d90c | ||
|  | bfd3edb617 | ||
|  | 56ba24962c | ||
|  | 19a2110ba2 | ||
|  | 242a3eec63 | ||
|  | fee46f2e54 | ||
|  | 6aed7e429a | ||
|  | 517ea82b99 | ||
|  | 99c29343de | ||
|  | 892fa76883 | ||
|  | d446b57d05 | ||
|  | 0e086d788b | ||
|  | 498841d45d | ||
|  | d1f8ee1e56 | ||
|  | 07feb5c925 | ||
|  | 75fd263e85 | ||
|  | 1e1f444cab | ||
|  | 89cc7e5fa9 | ||
|  | 265e7aefbf | ||
|  | 1c55a6c6dc | ||
|  | 8f18b5b8a7 | ||
|  | f790182f0b | ||
|  | 813007a5d8 | ||
|  | d03ff1e4d0 | ||
|  | 932bbd0381 | ||
|  | 01bd648cb2 | ||
|  | 779a5606a7 | ||
|  | ccc11e49d2 | ||
|  | d28c994ecd | ||
|  | 5d88717f32 | ||
|  | e35cfd4971 | ||
|  | bcc4bf5c2b | ||
|  | a0594cbce3 | ||
|  | 078bfefe41 | ||
|  | 9c1b11d605 | ||
|  | 44d82f9190 | ||
|  | 37fcde30d6 | ||
|  | 09c6cb4d6b | ||
|  | b428343c2a | ||
|  | dfce292294 | ||
|  | 2b8f613a00 | ||
|  | 2eb137618e | ||
|  | 4bb2b8ca9b | ||
|  | 5179562fb2 | ||
|  | 0a4de45453 | ||
|  | ffdc658dc8 | ||
|  | 7530f4407b | ||
|  | 73864c8101 | ||
|  | f948917124 | ||
|  | 0d44492086 | ||
|  | 38a22fbc99 | ||
|  | 8ae435549d | ||
|  | 9b113c0cbb | ||
|  | 0e0fac8c4b | ||
|  | 4cd9bb8f99 | ||
|  | ad9eaeafeb | ||
|  | 6cd392909c | ||
|  | 49ec430592 | ||
|  | 09f3fbeb38 | ||
|  | e7698686fa | ||
|  | 66d939df0d | ||
|  | 6bc079c7b7 | ||
|  | 299419917e | ||
|  | 69f6afe420 | ||
|  | b7279a3d9e | ||
|  | e14b854d7b | ||
|  | 8bd7c601c0 | ||
|  | 997288fa03 | ||
|  | 0f26b39997 | ||
|  | ae66fcac1e | ||
|  | 43944a94eb | ||
|  | eba0bde6f3 | ||
|  | 4544af441b | ||
|  | a8be94de6b | ||
|  | b24df31c2b | ||
|  | 332ba8ed7e | ||
|  | 58400f53bc | ||
|  | 01c2112881 | ||
|  | a546c2247d | ||
|  | 0da9142009 | ||
|  | 796add0ee2 | ||
|  | 00b32f64e6 | ||
|  | f97b3f23e2 | ||
|  | 08a079a96e | ||
|  | e98e951834 | ||
|  | 2668bf8519 | ||
|  | dd4c073e18 | ||
|  | c7c72f00c7 | ||
|  | ef1c665b9a | ||
|  | d56565be25 | ||
|  | e076b3aedc | ||
|  | ae3b2ddf5f | ||
|  | 1bdc427d73 | ||
|  | 6a639ce533 | ||
|  | d91ca8b197 | ||
|  | a01c0a283d | ||
|  | 5c393091a0 | ||
|  | 01b680504b | ||
|  | 8e4319cd5a | ||
|  | 5a776dd690 | ||
|  | cce08d95db | ||
|  | 28c1b208c1 | ||
|  | 3844bf1f72 | ||
|  | 745d192563 | ||
|  | ee782e92ac | ||
|  | afbc91d1fc | ||
|  | f998888d6d | ||
|  | 7d8b42d63e | ||
|  | 6ebd4e821f | ||
|  | d1806bfdc3 | ||
|  | 1d2d7155da | ||
|  | b09e0a05bf | ||
|  | c609e982fe | ||
|  | 2b227b43fe | ||
|  | 48f09f71ab | ||
|  | ead8c59bda | ||
|  | db52a9466c | ||
|  | 1509de390e | ||
|  | 88a1aa4f3d | ||
|  | 172e78e8f2 | ||
|  | 36bfef567d | ||
|  | e40ebd75a2 | ||
|  | 992732f2cb | ||
|  | b58a3ba1bb | ||
|  | afe521b0c9 | ||
|  | 5d9caef45f | ||
|  | 278e2f5605 | ||
|  | 1e299bf360 | ||
|  | 8dfa0bc38c | ||
|  | fde136fb7b | ||
|  | ee4da1a757 | ||
|  | ae2d96c455 | ||
|  | 6d8fbe0877 | ||
|  | 2fa1d8f2e8 | ||
|  | 533090a68e | ||
|  | 1dff59e1d6 | ||
|  | 44d232f52a | ||
|  | 5f6cff739a | ||
|  | 2764d235a9 | ||
|  | 45debff89f | ||
|  | c45fbe6310 | ||
|  | 9ef9c24388 | ||
|  | 6a40f23578 | ||
|  | 6a0a6b4751 | ||
|  | 0bee6f6b41 | ||
|  | 82a15b5a16 | ||
|  | 11b7c4459e | ||
|  | 98570ac456 | ||
|  | 1b2296ad5b | ||
|  | 16851746d6 | ||
|  | 935450a45f | ||
|  | ba67fd318b | ||
|  | 08ac459a41 | ||
|  | a83e9d9a0a | ||
|  | 62d3f01948 | ||
|  | af5ca2d0b8 | ||
|  | ab4bcdf12d | ||
|  | a6756d2cea | ||
|  | f81061dd42 | ||
|  | 8e2c304b3c | ||
|  | f21adaa3ef | ||
|  | 2637939e62 | ||
|  | faf05582f8 | ||
|  | 161c02ced3 | ||
|  | ff8de8e42d | ||
|  | 09d506194f | ||
|  | 42db3085df | ||
|  | ad14c88fde | ||
|  | 0c9daf6eaf | ||
|  | 86c6530e46 | ||
|  | 159f80d629 | ||
|  | aa949165c7 | ||
|  | d22359b6e7 | ||
|  | d73709653d | ||
|  | 405926e811 | ||
|  | 36758f41a4 | ||
|  | 7ebc9c79cf | ||
|  | e0668b55b9 | ||
|  | 76c09da961 | ||
|  | 7e3b8c2c59 | ||
|  | ecca854c7c | ||
|  | 3b0d7ea960 | ||
|  | f70fa42eac | ||
|  | 5698de6cf4 | ||
|  | c5a333a904 | ||
|  | ff324955dd | ||
|  | 70436f5dca | ||
|  | 31177a2b1b | ||
|  | 4de012fc49 | ||
|  | ee2888e744 | ||
|  | efe4df92dc | ||
|  | 723ab54f97 | ||
|  | d9389afc66 | ||
|  | e7178ee496 | ||
|  | d5f35bb3fb | ||
|  | 72f1a779f2 | ||
|  | 3277544295 | ||
|  | 98d2c64d5d | ||
|  | f68b46fc60 | ||
|  | d54ab856e7 | ||
|  | 16b24fadea | ||
|  | b3803cbdf1 | ||
|  | 2ceaa25181 | ||
|  | 513611c5a6 | ||
|  | 7ec4ba40ad | ||
|  | 92374e122b | ||
|  | 94f12732ab | ||
|  | 0904712a00 | ||
|  | 32becdbced | ||
|  | 34aa21f7d9 | ||
|  | cc81dd7d3e | ||
|  | 335213b55f | ||
|  | 13ab4166c0 | ||
|  | 3dc5a0e7f8 | ||
|  | e15c5cde53 | ||
|  | d88c09b098 | ||
|  | 893b383bdf | ||
|  | dd7c9d62e6 | ||
|  | 97c5c90eff | ||
|  | 1fb94e7a7b | ||
|  | daca87c6d0 | ||
|  | 203ec5fa46 | ||
|  | 9ea69c07b8 | ||
|  | 68539d6cc9 | ||
|  | f75fd0811e | ||
|  | 836bc9d456 | ||
|  | a37769aafe | ||
|  | 68e62e4bd2 | ||
|  | a5cd3728c9 | ||
|  | a48ce35f0b | ||
|  | e1835b5775 | ||
|  | 433832b329 | ||
|  | ee81da14d6 | ||
|  | 6395d1908e | ||
|  | 989a5a2f8a | ||
|  | b7a622c68e | ||
|  | a8507b437d | ||
|  | e505bf9ccf | ||
|  | a289b32053 | ||
|  | c3f1f09ad1 | ||
|  | 70ee2026ff | ||
|  | 690782bf60 | ||
|  | 755cc4835e | ||
|  | a684ea46e4 | ||
|  | 8fbe13f99d | ||
|  | 452e9e275f | ||
|  | cd40088636 | ||
|  | 9b9e6f4af5 | ||
|  | ae6eeadf54 | ||
|  | 5268b05060 | ||
|  | 390263a34e | ||
|  | 55646edc3e | ||
|  | 8d177beb78 | ||
|  | 1da0c59182 | ||
|  | 36e8f10d2b | ||
|  | cdf5a8f20f | ||
|  | eb64d92333 | ||
|  | eb55da63ef | ||
|  | 918302f79b | ||
|  | 9d7131d9f6 | ||
|  | 229c1114dd | ||
|  | 885df9156f | ||
|  | c319233ddc | ||
|  | 958b5c0780 | ||
|  | 880c0a5da8 | ||
|  | 237c6dc856 | ||
|  | ccf6e32bf9 | ||
|  | a1874f6f00 | ||
|  | 95e4490a8a | ||
|  | 31c132c2eb | ||
|  | 00b0ec58b4 | ||
|  | a1d0e5bb65 | ||
|  | 03e0d4b2e8 | ||
|  | 6afdd4e6fd | ||
|  | b500a0d477 | ||
|  | dd2463a440 | ||
|  | 23a8bebd9e | ||
|  | 3caf9108ad | ||
|  | bde4be8231 | ||
|  | 0bbbb12ed2 | ||
|  | b570bdaed7 | ||
|  | 8c0843cc87 | ||
|  | 31458ffd81 | ||
|  | c15c10a94e | ||
|  | 9fca978725 | ||
|  | b125901717 | ||
|  | eb018ae660 | ||
|  | 7e5a9474fe | ||
|  | 525a9b5036 | ||
|  | c3fbdf34ca | ||
|  | 48bd51e1a5 | ||
|  | 10d0b03a90 | ||
|  | e1b3582f08 | ||
|  | 95be1c9e22 | ||
|  | 1ce8fe06d5 | ||
|  | 15c649024e | ||
|  | e97303c226 | ||
|  | 3b786c819d | ||
|  | 04959dbd8b | ||
|  | 5cd4b874ea | ||
|  | f14ea1b3de | ||
|  | 9cc0cda0fb | ||
|  | 09a7a4bbe5 | ||
|  | cfea8b3745 | ||
|  | 28bf0b61ce | ||
|  | 2dc2429735 | ||
|  | 83d4592526 | ||
|  | 2d528c26ae | ||
|  | 66b3dce794 | ||
|  | 93f77a1045 | ||
|  | aa4d23a3d5 | ||
|  | 2d7ebff8e9 | ||
|  | bad9dd3b3b | ||
|  | 2f4e517857 | ||
|  | ff35ba3696 | ||
|  | 72768e7fad | ||
|  | 77f3852cdc | ||
|  | 66857ca477 | ||
|  | 75514fc7af | ||
|  | be06d871b6 | ||
|  | f98ee326b4 | ||
|  | bc8126eb16 | ||
|  | 4c8beefdcb | ||
|  | bbb6c53457 | ||
|  | d8991894e3 | ||
|  | c7b7dcfd03 | ||
|  | 2c9e50873c | ||
|  | 923367296d | ||
|  | 151a206617 | ||
|  | e403c4cf99 | ||
|  | e3fbe37f9f | ||
|  | dc870cd5ea | ||
|  | 584be44743 | ||
|  | 5fffd35ec1 | ||
|  | b92e22e4a6 | ||
|  | 3e6d16a7a8 | ||
|  | ecbcc277b8 | ||
|  | dff1d9e4dd | ||
|  | 7c0bde7310 | ||
|  | a82d21ac05 | ||
|  | 0bf8378fcb | ||
|  | 017ef8a837 | ||
|  | 0d63cdcb96 | ||
|  | 68a6f99c9f | ||
|  | 60781bcfc4 | ||
|  | 77fa2e2722 | ||
|  | c36afd872e | ||
|  | 7e58a4c130 | ||
|  | 19a4bf1088 | ||
|  | 9678bbae4b | ||
|  | a4d093afa1 | ||
|  | ba788bcf0f | ||
|  | f2c62bee7e | ||
|  | 548721e306 | ||
|  | 1ae950a638 | ||
|  | c9385e93fe | ||
|  | 9bb16e293c | ||
|  | c223702ea0 | ||
|  | 9167ba499d | ||
|  | 2d7e95e1b6 | ||
|  | 0cba736446 | ||
|  | 0816a57032 | ||
|  | a0ab0bd3e2 | ||
|  | b89ad4b328 | ||
|  | 6cda76a116 | ||
|  | c112b327ab | ||
|  | 46c12a8899 | ||
|  | c5219dfb3f | ||
|  | 4a8ee6815a | ||
|  | e1b6bb154a | ||
|  | b19c282269 | ||
|  | e520921746 | ||
|  | 970642244b | ||
|  | 3b90be2d9e | ||
|  | 2f756f1e3a | ||
|  | 78e84182f0 | ||
|  | 65a7a8caf8 | ||
|  | 4c6a2f5df9 | ||
|  | fea297e409 | ||
|  | 7cf6aba625 | ||
|  | 3bbc00cc8c | ||
|  | 70ed2b4203 | ||
|  | 0adce9b9c6 | ||
|  | 0e781d18fa | ||
|  | 4575a8fffe | ||
|  | 10d0ff252b | ||
|  | c7d54570cc | ||
|  | 7136b33f2e | ||
|  | 70a78e74f6 | ||
|  | d5707b7bf3 | ||
|  | 9f247901d4 | ||
|  | 5659742d97 | ||
|  | 450eaf7c4a | ||
|  | 47485e4b49 | ||
|  | 64254e758d | ||
|  | c1aa5d4e47 | ||
|  | ab8173637a | ||
|  | 3841cef497 | ||
|  | b717f1c7eb | ||
|  | da57f76de3 | ||
|  | 4784f1c65a | ||
|  | 41af63b333 | ||
|  | e2bb0de24d | ||
|  | b791fae9ce | ||
|  | 6033a9e20c | ||
|  | 9e8c8973d8 | ||
|  | 3933bf5c1a | ||
|  | 708e296774 | ||
|  | 84925ab69c | ||
|  | b3cb9b7fe2 | ||
|  | 9cb61fa34d | ||
|  | 7c219d235c | ||
|  | 6938c79f88 | ||
|  | b8284a147d | ||
|  | 15ee90e99c | ||
|  | 795f80b4ec | ||
|  | 6b6427492d | ||
|  | 6055b8c3dc | ||
|  | a98cb50d55 | ||
|  | e98bbc1c52 | ||
|  | 7245aece4f | ||
|  | 60cbb02822 | ||
|  | 4e863ecdac | ||
|  | 5037033fcf | ||
|  | e6b158bc97 | ||
|  | 4cc0dfa10b | ||
|  | 4ced8889d3 | ||
|  | d26967a87d | ||
|  | fc8955941b | ||
|  | 071a80360f | ||
|  | d2154f5f2e | ||
|  | 334d382bfa | ||
|  | 90c4b00f74 | ||
|  | 71261525e8 | ||
|  | 3126959576 | ||
|  | 02e51d8282 | ||
|  | ffb2027a19 | ||
|  | 70c9ab9074 | ||
|  | 6d1fdf1ba6 | ||
|  | 1f7180d9a8 | ||
|  | b4e94ae4dd | ||
|  | 07c606bfc9 | ||
|  | e705a8bd89 | ||
|  | b3bdfb7f1f | ||
|  | 5af1aeb092 | ||
|  | be64fa674a | ||
|  | 204f5591a9 | ||
|  | ee3e3a3a40 | ||
|  | f9200a2b75 | ||
|  | f570b70827 | ||
|  | 0db141eeac | ||
|  | acb2ee53bb | ||
|  | c544b7f5ba | ||
|  | c0024e97e5 | ||
|  | bdf8aa9168 | ||
|  | de5ce0f515 | ||
|  | bb95484c8a | ||
|  | cad18b8a3a | ||
|  | 0f6a98751a | ||
|  | aac5a4c27f | ||
|  | d3f6415387 | ||
|  | 04da44eb98 | ||
|  | 7649be97b1 | ||
|  | c9ef777e0f | ||
|  | c0cb2438d5 | ||
|  | 30c531b39e | ||
|  | bf703a8a66 | ||
|  | e7b631b087 | ||
|  | a9f5dc036c | ||
|  | 0a83b51e00 | ||
|  | eab63ecc6c | ||
|  | b0794cf35e | ||
|  | 5b9e71a27d | ||
|  | eae41de27d | ||
|  | e9163aa3a7 | ||
|  | 8c617515ba | ||
|  | 04e4e71f2e | ||
|  | a587482edf | ||
|  | 0aac9350d5 | ||
|  | f56c12ee4e | ||
|  | 4bb9ae61f2 | ||
|  | ff7f3484e4 | ||
|  | 5da3abe6b4 | ||
|  | c0b398e0ce | ||
|  | 3de10adac2 | ||
|  | 1b573d6552 | ||
|  | 2a96f93919 | ||
|  | c6b2639ca4 | ||
|  | b9abf37a7e | ||
|  | 373cbb4144 | ||
|  | a521982576 | ||
|  | a77fde577c | ||
|  | ea6926e57d | ||
|  | ba25b7fee6 | ||
|  | 7ee162d98b | ||
|  | 380f557c45 | ||
|  | 1bdae53f4e | ||
|  | 9314c346da | ||
|  | bfaad1388c | ||
|  | 0b580ad05d | ||
|  | bb35a80177 | ||
|  | 24fc95ac81 | ||
|  | 8f864417c4 | ||
|  | bb9d29b061 | ||
|  | b9d8ec1463 | ||
|  | 1842a7660d | ||
|  | 5caa2f5536 | ||
|  | d6078be8b7 | ||
|  | cf60723f14 | ||
|  | f7ff0a2b1d | ||
|  | cc49664b2f | ||
|  | 99fe74f026 | ||
|  | b021869eeb | ||
|  | b8806d163b | ||
|  | 1116aae1de | ||
|  | 5e5f60253b | ||
|  | bbc02752c9 | ||
|  | 9896bc110e | ||
|  | ca60f8ecdd | ||
|  | 544acd1e35 | ||
|  | 6e07602d77 | ||
|  | 82898f7bba | ||
|  | d61283a8bc | ||
|  | 1ee3f826cc | ||
|  | 4a00a5ba9e | ||
|  | 39eda67867 | ||
|  | a99d38fdaa | ||
|  | 0eb2d437e2 | ||
|  | 3ac9036c79 | ||
|  | c94e292176 | ||
|  | 91d87c2d9b | ||
|  | ff472f69c0 | ||
|  | e18119e24c | ||
|  | 4a592dc64c | ||
|  | d9e13201dd | ||
|  | 5c75b19c5d | ||
|  | 52a77db60f | ||
|  | 0513c250fb | ||
|  | 48864ad6cf | ||
|  | cdbccad21e | ||
|  | e15bc68c9b | ||
|  | 8bffd7672d | ||
|  | 61df5b3060 | ||
|  | b5255444cd | ||
|  | 0c94e377fc | ||
|  | 8e5c67b4b2 | ||
|  | b24f2f1756 | ||
|  | c69c17de42 | ||
|  | 061617122a | ||
|  | 125ce3240f | ||
|  | 7215efe167 | ||
|  | 06d1570142 | ||
|  | 093c370faa | ||
|  | aec9574737 | ||
|  | 7ceb76cff5 | ||
|  | 300e2fe9f8 | ||
|  | 91e1643627 | ||
|  | 91421b0c62 | ||
|  | 40f611664f | ||
|  | dcba4f4098 | ||
|  | c098ad2b3b | ||
|  | b43223cb7a | ||
|  | e243531dab | ||
|  | 1af38e62bc | ||
|  | f37f062cdc | ||
|  | 7e734214dc | ||
|  | 05d152746f | ||
|  | dea7f37553 | ||
|  | 415c599310 | ||
|  | 70cd4fedbe | ||
|  | 1e6d7673bc | ||
|  | b4963b725b | ||
|  | 0371ffa4ce | ||
|  | 6a664a7e15 | ||
|  | 88ce9300bc | ||
|  | 85cf0e311c | ||
|  | 0e3d75cfeb | ||
|  | 630c8a5faa | ||
|  | 905921a684 | ||
|  | 1e469b3b0f | ||
|  | bff3c4f95c | ||
|  | bd2bcb6994 | ||
|  | 4c8898a639 | ||
|  | 97df33ab1a | ||
|  | ef46fb2685 | ||
|  | d5d6dd3614 | ||
|  | 6c233c6a0a | ||
|  | 6db715d879 | ||
|  | ab02e8a546 | ||
|  | 8cbfe64f19 | ||
|  | fd1e9971e4 | ||
|  | 68336a76c5 | ||
|  | 393e914a86 | ||
|  | ffb54110e9 | ||
|  | 533d825f1a | ||
|  | c65279b672 | ||
|  | f9926beeef | ||
|  | add8a777d8 | ||
|  | 21bc505d85 | ||
|  | 3fc49c001e | ||
|  | 3d69a95c49 | ||
|  | d81fdf6d6b | ||
|  | 87d3109ffb | ||
|  | 180dbbb521 | ||
|  | 24aac7cee5 | ||
|  | 53e18a5387 | ||
|  | 92062d056d | ||
|  | 06368ab0a1 | ||
|  | 38efe25c68 | ||
|  | 319079de7a | ||
|  | 025bf900a5 | ||
|  | 2885f4f7b1 | ||
|  | c07eda15b1 | ||
|  | 4274296cf3 | ||
|  | 76a203d4df | ||
|  | 24f37e2062 | ||
|  | f465b2e2a0 | ||
|  | ce00e49a89 | ||
|  | d494f9d66b | ||
|  | c35a183a64 | ||
|  | 9cdd5fe7f2 | ||
|  | c21428215e | ||
|  | 64d5af46f5 | ||
|  | 25846ea18a | ||
|  | 798383596d | ||
|  | 9ca71bc937 | ||
|  | 5407429ec0 | ||
|  | ee5c94f6db | ||
|  | 91045afbee | ||
|  | 3f64782023 | ||
|  | f8d35f9502 | ||
|  | ea78d3ec9a | ||
|  | e056a28316 | ||
|  | 0bea721c2e | ||
|  | e1b89494d0 | ||
|  | cd8e7f3912 | ||
|  | 50604c25c2 | ||
|  | aa6b2357d8 | ||
|  | 5b2d29bef6 | ||
|  | a296d26328 | ||
|  | d01a26ec61 | ||
|  | efd7d6f0c0 | ||
|  | b55be093be | ||
|  | 7c1d5cadd7 | ||
|  | dd1592b03b | ||
|  | 9b37ac483f | ||
|  | 090820958e | ||
|  | ac21e1be5c | ||
|  | 5196443b26 | ||
|  | c8531cbeb1 | ||
|  | c560abedba | ||
|  | 9b952fbc44 | ||
|  | ccdf05e922 | ||
|  | c3d74f2ae9 | ||
|  | f47498888c | ||
|  | 5665a7f0cb | ||
|  | b8178c6c8d | ||
|  | 4a0f15eb88 | ||
|  | c4f53fe525 | ||
|  | 8c93ec52de | ||
|  | befe0fff2a | ||
|  | b6a837cbea | ||
|  | 4861973899 | ||
|  | c593e4b500 | ||
|  | 5bf78c20d4 | ||
|  | 5c672130e6 | ||
|  | d8214d4f12 | ||
|  | 64d1f09ce0 | ||
|  | 47d0f0ea40 | ||
|  | 2d85fd093e | ||
|  | d936568b76 | ||
|  | 4598a83e8e | ||
|  | f4bf00ad31 | ||
|  | 07fde7f6cc | ||
|  | 729209574e | ||
|  | f28206d989 | ||
|  | 0c81b32cac | ||
|  | 11216017cb | ||
|  | a7b9f53967 | ||
|  | 1fa2e2e37d | ||
|  | f67d5faeb7 | ||
|  | 5cbf859458 | ||
|  | 629ed74d09 | ||
|  | ca2af2ca63 | ||
|  | 52ab089615 | ||
|  | 01461a196d | ||
|  | 04832f052a | ||
|  | c8b2c8ae50 | ||
|  | 1b81c7fb22 | ||
|  | 9ccda0247e | ||
|  | a7df4dcf25 | ||
|  | d91f47c791 | ||
|  | a9ac4e7f44 | ||
|  | fc3ec57437 | ||
|  | 266f6ab919 | ||
|  | 6218c1c00b | ||
|  | cc81d6fe82 | ||
|  | 69f9102f2d | ||
|  | beb9275982 | ||
|  | abe48713f2 | ||
|  | 82cfaf2fbb | ||
|  | 3d3bc4738f | ||
|  | 2d0746f5a4 | ||
|  | 9c71e2f1c8 | ||
|  | 134fd62da8 | ||
|  | 2afd283582 | ||
|  | c66734bab0 | ||
|  | 8e56a61f95 | ||
|  | d265271148 | ||
|  | b40e397b28 | ||
|  | 35ff1d996a | ||
|  | deea0b05cb | ||
|  | c50c9ca545 | ||
|  | a819b4a5a5 | ||
|  | df2d7d4734 | ||
|  | 79ce4098cf | ||
|  | 374464a1f8 | ||
|  | c8d0bf27af | ||
|  | 6e4ae034b2 | ||
|  | 52b560e72d | ||
|  | 9b971ad222 | ||
|  | 3613162d09 | ||
|  | 3a272e998d | ||
|  | d4c750beb4 | ||
|  | 84b31e65e1 | ||
|  | 7b802bfd3d | ||
|  | f9c4632b8d | ||
|  | e4764cd8a6 | ||
|  | dd78a3a686 | ||
|  | 94c06e13f4 | ||
|  | e8bebe5a75 | ||
|  | 5b0e1b4f9e | ||
|  | 8c0a93779b | ||
|  | 9241479da4 | ||
|  | 8ffca93cd5 | ||
|  | 7fea0c124a | ||
|  | 20dbdb20d2 | ||
|  | e6b8e2e8be | ||
|  | 7c5b7f77cc | ||
|  | de84547a21 | ||
|  | 44676756ae | ||
|  | b399b0f182 | ||
|  | 1152191f48 | ||
|  | af1b07ad44 | ||
|  | b8113fff1e | ||
|  | ff6948cf2d | ||
|  | fd25e85d59 | ||
|  | c07cd72e85 | ||
|  | e2c101206c | ||
|  | 92276b5769 | ||
|  | a2133f61a8 | ||
|  | 199adbbcf0 | ||
|  | dc316fd7b4 | ||
|  | 025183602f | ||
|  | db4619a9d9 | ||
|  | 451e527b7c | ||
|  | 54dd3a00df | ||
|  | 03c5dab79d | ||
|  | 1fdee861e8 | ||
|  | c12bf991b3 | ||
|  | 78a097585d | ||
|  | 39132327cc | ||
|  | dc32318cec | ||
|  | 592f74124c | ||
|  | e5e63cc5ac | ||
|  | f40e0f786d | ||
|  | ebd9f1471b | ||
|  | d76547ead4 | ||
|  | 4600772e05 | ||
|  | ed597423cd | ||
|  | f20ca06f85 | ||
|  | a636d3f394 | ||
|  | 043df18daa | ||
|  | 96996bf18e | ||
|  | f350137a14 | ||
|  | b7a6f3ec75 | ||
|  | 6c34672549 | ||
|  | e779a07bce | ||
|  | 9a36e8ba3b | ||
|  | c968bacb01 | ||
|  | 25199dfb43 | ||
|  | 48fed4e6fb | ||
|  | fc253237c9 | ||
|  | 589948c7f4 | ||
|  | 7e69690605 | ||
|  | 95f498ba9b | ||
|  | fd07ae5225 | ||
|  | 8acd94fc89 | ||
|  | 1436480eab | ||
|  | 448d176c24 | ||
|  | fd269453a4 | ||
|  | b3b380964c | ||
|  | 6e9025ebf2 | ||
|  | 3922691b3c | ||
|  | 0545b77cf4 | ||
|  | 6b3f39fa1a | ||
|  | 3114ab87dc | ||
|  | 00bc99cc7b | ||
|  | 540b3ae2f4 | ||
|  | dbfe4140e1 | ||
|  | d3675ec254 | ||
|  | ded2483fc0 | ||
|  | e62ea388e0 | ||
|  | f20356e9be | ||
|  | d282a2d846 | ||
|  | 4641ac46e7 | ||
|  | ba9268a09e | ||
|  | fb9902c536 | ||
|  | 5318ba6c6e | ||
|  | fd5ebef488 | ||
|  | d9e4f39ddc | ||
|  | 435b9d8973 | ||
|  | 0ea70ba656 | ||
|  | 92a07b87d2 | ||
|  | c3c82282ba | ||
|  | adc15c24ef | ||
|  | dddf9a9396 | ||
|  | 9ca6860ffa | ||
|  | f7dd388954 | ||
|  | 6012839f0e | ||
|  | 8e9cbab053 | ||
|  | aaf375a57b | ||
|  | 3cce985f03 | ||
|  | c59df6ec20 | ||
|  | 5c3f41f64d | ||
|  | cf3523f49f | ||
|  | db794752cb | ||
|  | bceaebe856 | ||
|  | 3916de2921 | ||
|  | 9e0f8c1a97 | ||
|  | 0cbc56b82e | ||
|  | b95608f68a | ||
|  | b6e5dbd06c | ||
|  | 914f19be86 | ||
|  | f09bcf3fcf | ||
|  | d0b18dec8e | ||
|  | 75d486b124 | ||
|  | 4914609485 | ||
|  | 75bd66326a | ||
|  | 8f904f75bb | ||
|  | 549c598f51 | ||
|  | ed68d604d6 | ||
|  | f83752f43b | ||
|  | 86c22636eb | ||
|  | 30d20a453b | ||
|  | fe29d8a23f | ||
|  | 694d088160 | ||
|  | 6aabbffc62 | ||
|  | 7b59bc8d12 | ||
|  | 79d0fb0b52 | ||
|  | edf56d34f8 | ||
|  | 623329fb33 | ||
|  | 9f0074eef9 | ||
|  | 6733253826 | ||
|  | f117805129 | ||
|  | c75b1581d2 | ||
|  | 109e118aba | ||
|  | 201b77d5b6 | ||
|  | a5ca08f33d | ||
|  | 86210c4513 | ||
|  | 988a3e4446 | ||
|  | 0f5cd22bb7 | ||
|  | 2f5bed36b3 | ||
|  | 5b6534bb28 | ||
|  | e31e5b2477 | ||
|  | 07d5fafe2e | ||
|  | e08da659e5 | ||
|  | 8a4979f44c | ||
|  | e67464325f | ||
|  | 94c9b0d23b | ||
|  | e9ec310d8a | ||
|  | c78d1e3c39 | ||
|  | e94319145f | ||
|  | 3f3b01b5f6 | ||
|  | 19a2791c65 | ||
|  | 4e8ccf0ef3 | ||
|  | f1a7d5ecf7 | ||
|  | 8b05abb80d | ||
|  | 48c9349ce9 | ||
|  | 117d848466 | ||
|  | 9a2df072cc | ||
|  | 99c62aab36 | ||
|  | 224278e07a | ||
|  | 74b69e191e | ||
|  | 8cda8a727c | ||
|  | a3c0c7c96f | ||
|  | 4403e4ed62 | ||
|  | 9b209823f6 | ||
|  | b2cb125bd4 | ||
|  | 5e8f767642 | ||
|  | 6ee270d9d8 | ||
|  | 44fa309d20 | ||
|  | 58d88f3dd4 | ||
|  | e980c23177 | ||
|  | 75224321bb | ||
|  | 801af05b20 | ||
|  | 7611dbbddc | ||
|  | 6d40ca15bc | ||
|  | 32c1c19224 | ||
|  | bbf6357222 | ||
|  | dc16629c24 | ||
|  | 3718b9d768 | ||
|  | c25eb088ec | ||
|  | 3feb3e52f8 | ||
|  | 8e730ef93d | ||
|  | e0913a39ab | ||
|  | 7a27fbc001 | ||
|  | ee0dbdad35 | ||
|  | 9225f88f89 | ||
|  | a04839dd6b | ||
|  | 002006517a | ||
|  | f5b202d438 | ||
|  | a7df094ff4 | ||
|  | 1e6fa77633 | ||
|  | eb4cff202c | ||
|  | 7ee777f405 | ||
|  | 81bd5c784e | ||
|  | b526e132a7 | ||
|  | 1860f66de5 | ||
|  | ded9ada9bc | ||
|  | d0e6a2eb8b | ||
|  | 4e103a1963 | ||
|  | 475e927178 | ||
|  | ca7932c4f0 | ||
|  | 8ab47d3321 | ||
|  | def7e87151 | ||
|  | 27568c2bef | ||
|  | 0694a187d7 | ||
|  | 832601b36b | ||
|  | 578969c34c | ||
|  | d1d0115aed | ||
|  | c89e6ebfab | ||
|  | ca1089b881 | ||
|  | a1d04f2aad | ||
|  | bf0604133c | ||
|  | a82b2da16e | ||
|  | f2273c0acc | ||
|  | 17bedac96c | ||
|  | 4831fad27a | ||
|  | 5e896cf582 | ||
|  | add3491c57 | ||
|  | f470576822 | ||
|  | 10760a53a8 | ||
|  | eee805183c | ||
|  | b8fb391022 | ||
|  | 3c698f1584 | ||
|  | 2fad52d684 | ||
|  | ec64a68a71 | ||
|  | db55562f6a | ||
|  | d8409a9d2b | ||
|  | 0d0ce6eec1 | ||
|  | 483f313eda | ||
|  | 7b6c742178 | ||
|  | d4a35ba6ff | ||
|  | 68b112837a | ||
|  | e2f20ebf94 | ||
|  | f870e4965a | ||
|  | 7ebcb219d6 | ||
|  | c21913a66b | ||
|  | 77e956a29f | ||
|  | 08275c406a | ||
|  | 2931e1b87b | ||
|  | 153b422496 | ||
|  | 0f6a6d6fea | ||
|  | 91fdb3e2d4 | ||
|  | d8e87bd881 | ||
|  | 922033c1b2 | ||
|  | df1793efbf | ||
|  | 836a2700f2 | ||
|  | 8f3aaf77a1 | ||
|  | 00c059e5b1 | ||
|  | f4f355c74a | ||
|  | b465fc5aaf | ||
|  | 2d78eaa48d | ||
|  | d08451bccc | ||
|  | d8e785aed0 | ||
|  | 267b6f49b5 | ||
|  | e6688f4b9d | ||
|  | 9d7b9771c2 | ||
|  | 136a9a39de | ||
|  | 3dcf628fdb | ||
|  | e614e9787a | ||
|  | e426fc0922 | ||
|  | 5d4bfffc7e | ||
|  | 207cdaf7a4 | ||
|  | 7315b581ce | ||
|  | 38efaae7b2 | ||
|  | 469e042216 | ||
|  | 0f1a4b9d8f | ||
|  | 7303c00296 | ||
|  | fc55b34d84 | ||
|  | 6f67fc0e02 | ||
|  | 562d722ad5 | ||
|  | 144c1ba3a6 | ||
|  | 06b032af91 | ||
|  | 3603140114 | ||
|  | e094785cbd | ||
|  | e7408224ac | ||
|  | e67c05c274 | ||
|  | b22804efaf | ||
|  | 890f55f91a | ||
|  | cc5fc0b892 | ||
|  | 5efe2b027a | ||
|  | 5b6569d0f9 | ||
|  | 0eda7ac498 | ||
|  | a5ef353484 | ||
|  | 67a36d8d31 | ||
|  | 7cc3cc3990 | ||
|  | dc0edc4c2b | ||
|  | 71d2f091e5 | ||
|  | c2f062a391 | ||
|  | 224f490455 | ||
|  | 5b35232ab4 | ||
|  | 6d6db70e42 | ||
|  | 6830e15b4e | ||
|  | 3f07cad35d | ||
|  | e951340033 | ||
|  | db8912a735 | ||
|  | 0e297731a3 | ||
|  | f20c4f98ac | ||
|  | 05e60cc7c0 | ||
|  | 55b4469767 | ||
|  | f15516e478 | ||
|  | 17ceadbadf | ||
|  | 8c25b2b316 | ||
|  | 8b1ae404a3 | ||
|  | 13534cd4a9 | ||
|  | abfb345503 | ||
|  | 42ae935496 | ||
|  | 434515d957 | ||
|  | 094f7803b7 | ||
|  | b0c7bad391 | ||
|  | e9a4a905ef | ||
|  | 7b6cd0cfbe | ||
|  | b718b12083 | ||
|  | cfa7258ff4 | ||
|  | b70e0a0870 | ||
|  | da8eb464b8 | ||
|  | 8f9d1cfa30 | ||
|  | 585009ac5c | ||
|  | 30ee65fd14 | ||
|  | 76428b16f0 | ||
|  | 0d7b14e2d8 | ||
|  | a9d19d02b3 | ||
|  | adcbe55307 | ||
|  | aa99a7df64 | ||
|  | 00afa1ce52 | ||
|  | e94bf4c63c | ||
|  | ec5adffdc2 | ||
|  | 733c17ad3a | ||
|  | 53b0b562e6 | ||
|  | fabae6e970 | ||
|  | a9f9c40d8a | ||
|  | 6fc89607d3 | ||
|  | 2340760f53 | ||
|  | 39d6d2857e | ||
|  | 7b722a0001 | ||
|  | e7682119e0 | ||
|  | af6be44676 | ||
|  | 5a8f97a0b6 | ||
|  | 0d4dd385b8 | ||
|  | 94f0f3e966 | ||
|  | 43e31765e5 | ||
|  | 7c1bdfe713 | ||
|  | 9f09784b55 | ||
|  | e7a3a89bfb | ||
|  | 7ea7e63f44 | ||
|  | 1d2ce2cbeb | ||
|  | 06cf2e0bd7 | ||
|  | 9d219ae4b9 | ||
|  | 71f5a6c50e | ||
|  | 90b2be2bf4 | ||
|  | db1aa8fcbd | ||
|  | 11c000f764 | ||
|  | 4d6dcbd173 | ||
|  | 0da117efd2 | ||
|  | 533c368e32 | ||
|  | 8883513b0e | ||
|  | dcc9a71455 | ||
|  | 1a56743bb1 | ||
|  | 387a4b7c35 | ||
|  | 1d65d63bd9 | ||
|  | dda19c29fe | ||
|  | ca41669f4f | ||
|  | 0e1886e6bd | ||
|  | c26e116f0e | ||
|  | 5c9c7f2c5e | ||
|  | ca2fb6cef3 | ||
|  | 46dac909ef | ||
|  | b1e4347e10 | ||
|  | 97aa91c75e | ||
|  | 4f8fb32136 | ||
|  | e0fbce0087 | ||
|  | fb22f78fb3 | ||
|  | d6393cdbe5 | ||
|  | 5167fdb3f0 | ||
|  | ab00822764 | ||
|  | b4352ad38b | ||
|  | d07d00fa41 | ||
|  | 11d87e4725 | ||
|  | 627ed51a1b | ||
|  | c8f3bfa726 | ||
|  | 3091e3a1c8 | ||
|  | 2f3e7d1c27 | ||
|  | 0e831d4b92 | ||
|  | 7294ec9a3c | ||
|  | e34bab9585 | ||
|  | 7dd14955c1 | ||
|  | 6428ced157 | ||
|  | 30a42ec1bd | ||
|  | aacea3e9db | ||
|  | 6886b61186 | ||
|  | 0744c9fa29 | ||
|  | 502a665ffc | ||
|  | 3c315703c0 | ||
|  | 12ed07a607 | ||
|  | 101b33c381 | ||
|  | 97f4316653 | ||
|  | b0704e86f0 | ||
|  | a182b13e5a | ||
|  | 80b630a1e4 | ||
|  | 475efbe007 | ||
|  | 3ab5e5ac48 | ||
|  | c6c5ff2089 | ||
|  | 176ec8ac7d | ||
|  | dcdd4b3255 | ||
|  | fc0a0105b3 | ||
|  | f3960d21a8 | ||
|  | a44d853c1b | ||
|  | 6b41734d6a | ||
|  | c33dc0f3be | ||
|  | bb5ffb24a8 | ||
|  | a878c9a61d | ||
|  | 6454bf8ec4 | ||
|  | 40aa733ea7 | ||
|  | f37a822725 | ||
|  | f249ccd414 | ||
|  | 7ef4ddf0f3 | ||
|  | d8e18df3a1 | ||
|  | 78d3d9d27d | ||
|  | 0aa0ec5abd | ||
|  | b6eef3612f | ||
|  | 666d62dd7a | ||
|  | 44ee4b989f | ||
|  | 18790d867c | ||
|  | d6b8936376 | ||
|  | 4d840c7db8 | ||
|  | 4d2b21816d | ||
|  | 2d34fdd28f | ||
|  | 68abda1219 | ||
|  | f778f08f76 | ||
|  | ac1bd2fb7b | ||
|  | 4b7b1379d9 | ||
|  | e560e2ab3f | ||
|  | 1e441c2ddf | ||
|  | 93ce74eeb1 | ||
|  | f718f4251b | ||
|  | 4644c9b621 | ||
|  | 197081f10d | ||
|  | 00b717cde8 | ||
|  | 34aa917ca4 | ||
|  | a38ddcb364 | ||
|  | 5b9576df4e | ||
|  | 310219e5d7 | ||
|  | a0deb463c9 | ||
|  | 90ddec2ad8 | ||
|  | f6b03d5a78 | ||
|  | f531daa872 | ||
|  | 046dceb5c2 | ||
|  | dcc1f00048 | ||
|  | 05f935b598 | ||
|  | f2d27403c5 | ||
|  | 473efbe67a | ||
|  | aeabf0f324 | ||
|  | 80ab552ad8 | ||
|  | 7d4695c5b2 | ||
|  | 5189eaca36 | ||
|  | cfb31377fc | ||
|  | a07c52e112 | ||
|  | 8e1071aa89 | ||
|  | 7cb9a6ba60 | ||
|  | 350dc731f1 | ||
|  | f690f58bd4 | ||
|  | 4bc65e9ef7 | ||
|  | 2d600da8b6 | ||
|  | 35af53828a | ||
|  | 10ddd5b127 | ||
|  | f46e131f18 | ||
|  | feb5c8be95 | ||
|  | edf12bec71 | ||
|  | ff1fc28287 | ||
|  | 314398ba4c | ||
|  | 840331347b | ||
|  | 6181b12ab8 | ||
|  | 68da661edc | ||
|  | 88cbb6913d | ||
|  | 7a26646e1b | ||
|  | 92eb3b0bf6 | ||
|  | fb63434eee | ||
|  | 97f90d9684 | ||
|  | f91786367f | ||
|  | 6a57337a68 | ||
|  | 211e2bb37a | ||
|  | d2d08bf143 | ||
|  | 8acb37b6c2 | ||
|  | 81b3d2db4f | ||
|  | 9633c0b07a | ||
|  | 1dfa8ee7d8 | ||
|  | 1163543a98 | ||
|  | bdb7de34be | ||
|  | 9500fc11ac | ||
|  | 65daf29acd | ||
|  | 298b25cf7d | ||
|  | 41f4e22a17 | ||
|  | 288c57c144 | ||
|  | 7ff8923569 | ||
|  | b41779bd02 | ||
|  | beea6bc794 | ||
|  | fee58e98c5 | ||
|  | c51c1da618 | ||
|  | ea2812f50f | ||
|  | 3ec05709d5 | ||
|  | 4bdac7404a | ||
|  | cc41218d37 | ||
|  | 4b336b1853 | ||
|  | e1c77ce236 | ||
|  | 064d412ec8 | ||
|  | 7fff4f249d | ||
|  | 7a3745f642 | ||
|  | f8658f6afa | ||
|  | 223b725a10 | ||
|  | 25aad8d7be | ||
|  | b2c9b7635d | ||
|  | 24d13dd120 | ||
|  | 965340ff90 | ||
|  | 8e36fe6bef | ||
|  | 2eb41a8caf | ||
|  | fb989ae62f | ||
|  | 7901ec2a64 | ||
|  | f675dbc726 | ||
|  | 2ad4fdbbb9 | ||
|  | 97cb0cbd08 | ||
|  | 4ca0805de1 | ||
|  | 4b358abbb7 | ||
|  | dc82a0fc16 | ||
|  | 435d6f6f3f | ||
|  | ef92451d1a | ||
|  | 06184bdcb1 | ||
|  | af98d01053 | ||
|  | bb1cda0916 | ||
|  | a6d0ea347c | ||
|  | 0fcd57192b | ||
|  | a6ffa5738b | ||
|  | c75bd97537 | ||
|  | eea09f4de5 | ||
|  | 5656ec11d3 | ||
|  | eb53e44cb0 | ||
|  | 69f3106062 | ||
|  | 8ab99f6129 | ||
|  | 53a3c59a91 | ||
|  | df36983049 | ||
|  | bda016bb3b | ||
|  | cc174b7b85 | ||
|  | bf9d120081 | ||
|  | 775c85fc18 | ||
|  | 5a756aaed9 | ||
|  | dca092fd7c | ||
|  | c6e92ecac4 | ||
|  | 93008ff605 | ||
|  | 43c7b935df | ||
|  | 8f9a0a244a | ||
|  | fd13bd864e | ||
|  | 710f27afa9 | ||
|  | f537793b0b | ||
|  | f7183e38ee | ||
|  | 0a65dfdd10 | ||
|  | 3075578245 | ||
|  | b042b7705e | ||
|  | d56eb397f9 | ||
|  | 3054a1d32d | ||
|  | 0a3cd652b0 | ||
|  | f70b914779 | ||
|  | 46ca0ac10d | ||
|  | 031f647952 | ||
|  | 8f1c86f550 | ||
|  | 926fdecd13 | ||
|  | af2ca7a67e | ||
|  | 9e3e2ff81a | ||
|  | a9fe6472d9 | ||
|  | a862a81480 | ||
|  | dbb92881a1 | ||
|  | 10bf7f5d07 | ||
|  | 1e61d84fd1 | ||
|  | 8618ba1b60 | ||
|  | 3c8c44155d | ||
|  | 2002412026 | ||
|  | 7f69517fd4 | ||
|  | 851f8645b5 | ||
|  | c40cfaa388 | ||
|  | 0349d1d57c | ||
|  | 53049c02ee | ||
|  | 73a3a61729 | ||
|  | 5fe6aa2800 | ||
|  | c7eafd7c79 | ||
|  | 10b5fb5d72 | ||
|  | c4eaa944e2 | ||
|  | a735939d1e | ||
|  | 6ed5f04970 | ||
|  | b459b09b2f | ||
|  | 3f5877dbcc | ||
|  | e659b91c4d | ||
|  | e09f054058 | ||
|  | b646f50265 | ||
|  | 0a48ef3030 | ||
|  | ba614801ee | ||
|  | fd6eb47e68 | ||
|  | e69aeb8b98 | ||
|  | 26ea1da146 | ||
|  | c9e8c7a290 | ||
|  | 5e4eb92443 | ||
|  | 461b6499ef | ||
|  | c769920b6e | ||
|  | 181b98ef9e | ||
|  | 4e1184a400 | ||
|  | e52d9e3210 | ||
|  | dc6475c91b | ||
|  | 52f9956e92 | ||
|  | 0bf00d1ca4 | ||
|  | d1a707df57 | ||
|  | 4dc9b45297 | ||
|  | 6e31eebfb5 | ||
|  | a7df828932 | ||
|  | 517cf61d11 | ||
|  | 4be7bc8323 | ||
|  | 74c05d00a9 | ||
|  | 677613d30a | ||
|  | bacba629a5 | ||
|  | 14e36f1362 | ||
|  | d43ad849d1 | ||
|  | 627aa61184 | ||
|  | dad5b17ac8 | ||
|  | fef52c0112 | ||
|  | 8c4765b386 | ||
|  | 7c121bfc01 | ||
|  | 942c5cc04b | ||
|  | 348b3036ff | ||
|  | 09d3451d9d | ||
|  | b1a49e5f29 | ||
|  | da01a5b4dc | ||
|  | 3f9cdd9b56 | ||
|  | 0f9e87d7bb | ||
|  | 0869789214 | ||
|  | 10c8cc35c5 | ||
|  | 30c2e3e8ff | ||
|  | 86cc2f1075 | ||
|  | fa357a450b | ||
|  | b32641db87 | ||
|  | 0ee790969d | ||
|  | 7844ace934 | ||
|  | f4993d6e5d | ||
|  | 0fab806f36 | ||
|  | be2113d291 | ||
|  | 625d5b2313 | ||
|  | 6471c0c536 | ||
|  | 47c53fa60a | ||
|  | cf50e4f6ec | ||
|  | 7eea97d741 | ||
|  | 88b55ab93e | ||
|  | ee36d47c27 | ||
|  | 6f2fdbe447 | ||
|  | 0f36be0001 | ||
|  | 0f4a197e34 | ||
|  | 7dbff5b9e6 | ||
|  | 220246278a | ||
|  | 349e5a15e9 | ||
|  | bf7f4bba7b | ||
|  | ab1766a559 | ||
|  | 51bf33040a | ||
|  | a2c7273801 | ||
|  | ec6ac5bf24 | ||
|  | ec7501782d | ||
|  | 890b1c2d52 | ||
|  | c25d07259a | ||
|  | c960246eee | ||
|  | a01aee3111 | ||
|  | e2e951efdf | ||
|  | 3f6393f732 | ||
|  | b6eb343234 | ||
|  | 207a7e5160 | ||
|  | a0face4a28 | ||
|  | a8cf9f5cc4 | ||
|  | 461b38e653 | ||
|  | 8e4c0f7c22 | ||
|  | d78bfcc35c | ||
|  | 2b7c09e6ee | ||
|  | 036d9dbe59 | ||
|  | 1d342cc6af | ||
|  | 62b32b2211 | ||
|  | ae45ce517e | ||
|  | 5b3ccab7dc | ||
|  | 95f16c38a9 | ||
|  | d616cb283b | ||
|  | 9874fe2c23 | ||
|  | 520a142992 | ||
|  | 6ff56dc0bb | ||
|  | 1e63615592 | ||
|  | 3e62ffed0a | ||
|  | b133d51a83 | ||
|  | 037b89f018 | ||
|  | 20d06d9f9d | ||
|  | 156cf7315c | ||
|  | e2886e5303 | ||
|  | c6cf330e70 | ||
|  | 6be3b62d78 | ||
|  | c57af5e81b | ||
|  | f7431f809e | ||
|  | ea43c34de8 | ||
|  | fb6e9fa58f | ||
|  | b2ce1e8029 | ||
|  | d90c51220f | ||
|  | d1b14b68fa | ||
|  | d911728611 | ||
|  | 86a7200012 | ||
|  | 6ddb7453e1 | ||
|  | ad2355f8d3 | ||
|  | 582c498fe3 | ||
|  | 0a0c58d450 | ||
|  | 0dc592b819 | ||
|  | f46300016d | ||
|  | 3e1a7c6102 | ||
|  | f07065bf84 | ||
|  | 6d79903eb3 | ||
|  | e166329f34 | ||
|  | bb1bf6a88c | ||
|  | 30cbb6c9a8 | ||
|  | 4e33ab1e89 | ||
|  | 5494f309c0 | ||
|  | 3b6e7eccdd | ||
|  | e41d6787bb | ||
|  | ed30108961 | ||
|  | 12712ef812 | ||
|  | 0307f6b42c | ||
|  | 3e44620966 | ||
|  | 7424f1f768 | ||
|  | b5331d821c | ||
|  | 27f6d47efa | ||
|  | dbc7ad2ec4 | ||
|  | 7b27d270a2 | ||
|  | 97b3a0b093 | ||
|  | 06b38506d1 | ||
|  | fd581ffc37 | ||
|  | ff57c5e9d3 | ||
|  | 9b16d7c786 | ||
|  | 4c1bb18956 | ||
|  | 7d2bf892b1 | ||
|  | a99e77093f | ||
|  | 92737bb695 | ||
|  | 9b81955544 | ||
|  | 4a0031080a | ||
|  | 40e9fba312 | ||
|  | e227cc92ff | ||
|  | 73dbdbcbe6 | ||
|  | 3961f26635 | ||
|  | e51c274a18 | ||
|  | e75d0c58a9 | ||
|  | 9a798360f4 | ||
|  | 844ad09464 | ||
|  | 1e1d1efd90 | ||
|  | 240e6835c2 | ||
|  | 61398ee8f8 | ||
|  | e6e84859b7 | ||
|  | abcdd331db | ||
|  | 775d136b91 | ||
|  | dc93691fd9 | ||
|  | 48d782c69c | ||
|  | 0a04e626d7 | ||
|  | e7c4bf5ebf | ||
|  | 546a416f7e | ||
|  | 179a7a2792 | ||
|  | 251b6fcf70 | ||
|  | ab1fffb721 | ||
|  | da352a322c | ||
|  | 7d20458e82 | ||
|  | 5a54066f81 | ||
|  | a58e5a3399 | ||
|  | 9872f43cbf | ||
|  | 1078cc4642 | ||
|  | db7ae028b2 | ||
|  | 2b6f5dbd59 | ||
|  | f7aa0c45df | ||
|  | a72d58cdf9 | ||
|  | 067283834a | ||
|  | cf362c4a61 | ||
|  | 496245c801 | ||
|  | 859ab36347 | ||
|  | 1d740c7c36 | ||
|  | a03c4c3659 | ||
|  | 094ecceaac | ||
|  | 2812736ae5 | ||
|  | 6f87f8706c | ||
|  | 38beebe720 | ||
|  | fc1c3c6808 | ||
|  | 96ba895b84 | ||
|  | df35dfe3bf | ||
|  | c5504c6657 | ||
|  | 530e109433 | ||
|  | 6cce47b2f1 | ||
|  | 6185d5eca1 | ||
|  | 685ad1746e | ||
|  | 891f870ec0 | ||
|  | ad9933f0f6 | ||
|  | 1b86117754 | ||
|  | eeb3c968d6 | ||
|  | 406658a10f | ||
|  | 6a0551cea1 | ||
|  | 553f3b45d2 | ||
|  | 064a8e785c | ||
|  | 21e9723bb2 | ||
|  | 60b2c44a44 | ||
|  | c4fe3ecc0a | ||
|  | 2f18a8f6d0 | ||
|  | 5ac784e18a | ||
|  | 7a2164b4d0 | ||
|  | 0a43eae184 | ||
|  | 3117e2b2a3 | ||
|  | 41fece4643 | ||
|  | 7aa807ec7f | ||
|  | 4d16e1e14a | ||
|  | 73fc18099e | ||
|  | e34dac8dbb | ||
|  | af0e7f7187 | ||
|  | a3a6812608 | ||
|  | 2725c4ad4d | ||
|  | c8cd6e9460 | ||
|  | 0cd27d6129 | ||
|  | b47fc1c020 | ||
|  | de6ef7ef5e | ||
|  | f95fe8f1da | ||
|  | bd0dee5db5 | ||
|  | c13b7fd883 | ||
|  | f7e74b3088 | ||
|  | 343f01d5e1 | ||
|  | 08bacdd090 | ||
|  | 41b1c80492 | ||
|  | e5d7316e5d | ||
|  | b043c3a6da | ||
|  | 98b2855b9c | ||
|  | f3c52c409f | ||
|  | 1307bdc612 | ||
|  | 8c2e6971fc | ||
|  | 1903990f30 | ||
|  | 7d67005709 | ||
|  | 9acc2f92d1 | ||
|  | 72dfb0bda3 | ||
|  | 1635612430 | ||
|  | abda837d2f | ||
|  | 101fb0b8aa | ||
|  | 10de7dc1f9 | ||
|  | d2309b8114 | ||
|  | 6bdd81623f | ||
|  | f538c9f0c3 | ||
|  | 8ae3bad6f7 | ||
|  | 77de99b383 | ||
|  | 312949f336 | ||
|  | 1ab635bd7e | ||
|  | b35abd548c | ||
|  | 30e1c3307c | ||
|  | 08e052380a | ||
|  | 0e824c35cc | ||
|  | 548374ac2d | ||
|  | 9ad79fefc9 | ||
|  | 49d37c016e | ||
|  | 7c70c79a84 | ||
|  | 6916b8bff7 | ||
|  | 73dfb5f443 | ||
|  | 69b9dfa468 | ||
|  | ab61b8ba0a | ||
|  | 5c8c64242f | ||
|  | ddf96943f0 | ||
|  | e773be2f58 | ||
|  | f965804e6d | ||
|  | ec078eba72 | ||
|  | 1815cb1bc3 | ||
|  | 7b3cd71085 | ||
|  | 06128b5d07 | ||
|  | 990c8e1f18 | ||
|  | a170506356 | ||
|  | 5ecf2a3357 | ||
|  | fa48746ba9 | ||
|  | e2b8c069d7 | ||
|  | 14407bd1aa | ||
|  | 08db72903c | ||
|  | 46f9fab140 | ||
|  | b7d06f2c0a | ||
|  | 118196a0bf | ||
|  | 586ce1fc80 | ||
|  | adb979df38 | ||
|  | 3401cb5b4a | ||
|  | ebf1f12e97 | ||
|  | 5766208207 | ||
|  | 4bf4771f08 | ||
|  | 0e87db9eb7 | ||
|  | 1e053783f3 | ||
|  | 7afc96112b | ||
|  | 7bb41a30ed | ||
|  | 3d1b0eb843 | ||
|  | 5b9af0b5ae | ||
|  | 9219ec539d | ||
|  | c8bd57cd4d | ||
|  | 53bf8c09fd | ||
|  | 651c383668 | ||
|  | 9ed7587e3e | ||
|  | 674295e800 | ||
|  | 6b02f2eea0 | ||
|  | 5237e55326 | ||
|  | 3b59592110 | ||
|  | 72640ae058 | ||
|  | d916027e75 | ||
|  | 8966d2aa06 | ||
|  | de7ea04f54 | ||
|  | bf71fabe0e | ||
|  | b3368acb33 | ||
|  | 87220c6697 | ||
|  | a3b5c2ad71 | ||
|  | fb4c1473c5 | ||
|  | 2bb2502d20 | ||
|  | fe51698579 | ||
|  | 0f0f40bff3 | ||
|  | a798fe72d3 | ||
|  | fba98d03a5 | ||
|  | 7dd2517f67 | ||
|  | 641477d6f6 | ||
|  | 8e56656c8d | ||
|  | 564a6a1f62 | ||
|  | 69f0c80cd7 | ||
|  | 6fcb51cea2 | ||
|  | c58b8a4973 | ||
|  | c8f4ab4f06 | ||
|  | e425c4cca8 | ||
|  | 056ec986c2 | ||
|  | de3b2fb95b | ||
|  | 789e39c719 | ||
|  | b29c3152db | ||
|  | 3831679772 | ||
|  | 596f9566d8 | ||
|  | 124befe9d6 | ||
|  | 895534f32b | ||
|  | 50c16fe6de | ||
|  | b092d1a5d3 | ||
|  | a9b45630d7 | ||
|  | c1a39c269e | ||
|  | 6fa3f0b6cd | ||
|  | 9e5e3d1559 | ||
|  | 7135205299 | ||
|  | d99d977d2b | ||
|  | 7dd7e562bc | ||
|  | 75d857027e | ||
|  | 17694c1d01 | ||
|  | 749ad700d8 | ||
|  | 8f3df3039a | ||
|  | 02c315c194 | ||
|  | b697375573 | ||
|  | c57ef7725e | ||
|  | 3ae07503f2 | ||
|  | 9a0341adde | ||
|  | 96225efd96 | ||
|  | c3bd904f41 | ||
|  | 74257163b1 | ||
|  | 7bc75fd220 | ||
|  | a23281afab | ||
|  | 9e90dbdde6 | ||
|  | 1e8d8e40a2 | ||
|  | 583e208c1e | ||
|  | 9b91c427a1 | ||
|  | 196c5e9c24 | ||
|  | d8f7feb672 | ||
|  | 7c889f17b9 | ||
|  | c15a75556d | ||
|  | 5275c2e35f | ||
|  | 5267e06969 | ||
|  | b62183adcb | ||
|  | 5d2dec1803 | ||
|  | 4a98dab948 | ||
|  | 9f8c70b326 | ||
|  | a65404e63a | ||
|  | 05a1ddad05 | ||
|  | 4be3d63c0e | ||
|  | de6ce4a46e | ||
|  | 7a9e5afb93 | ||
|  | b2876b0a03 | ||
|  | b66f66fe6a | ||
|  | 30f04962d4 | ||
|  | 0feeb88024 | ||
|  | b799f2e05b | ||
|  | 56d21de001 | ||
|  | 7b54aa0c7d | ||
|  | 6e11b8ada1 | ||
|  | 98d25fc4e9 | ||
|  | 79405f47f6 | ||
|  | 1c7c4fc3b0 | ||
|  | 97e84d0977 | ||
|  | 9906b58818 | ||
|  | 371f084884 | ||
|  | 1c10839c14 | ||
|  | c55fdd9834 | ||
|  | 67b0890a6e | ||
|  | 4da4f96669 | ||
|  | 60a64c2558 | ||
|  | d4153da8b9 | ||
|  | fc33ab8905 | ||
|  | a123c64f59 | ||
|  | a090fe3834 | ||
|  | 8fa84de28e | ||
|  | 3e3da38de1 | ||
|  | a3be8ccc87 | ||
|  | cdfef30c22 | ||
|  | cabf1e82e8 | ||
|  | 608dc5e284 | ||
|  | 836d40072f | ||
|  | 431401d90e | ||
|  | 6da83e2bd7 | ||
|  | 7bccfc0006 | ||
|  | e051e09c1d | ||
|  | 1462c57d0c | ||
|  | 77c2b2b326 | ||
|  | 3cf9b9d9a5 | ||
|  | 629117e594 | ||
|  | 08f87c321f | ||
|  | 1ff13723fe | ||
|  | 510bda1b28 | ||
|  | 890327b381 | ||
|  | b21f7411dd | ||
|  | 5df623bd2e | ||
|  | 1e9d249f71 | ||
|  | a7b5949e6a | ||
|  | 02010170ce | 
							
								
								
									
										14
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| # These are supported funding model platforms | ||||
|  | ||||
| #github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2] | ||||
| #patreon: # Replace with a single Patreon username | ||||
| #open_collective: # Replace with a single Open Collective username | ||||
| ko_fi: irmen | ||||
| #tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel | ||||
| #community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry | ||||
| #liberapay: # Replace with a single Liberapay username | ||||
| #issuehunt: # Replace with a single IssueHunt username | ||||
| #otechie: # Replace with a single Otechie username | ||||
| #lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry | ||||
| #custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] | ||||
| custom: ['https://paypal.me/irmendejong'] | ||||
							
								
								
									
										41
									
								
								.github/workflows/all-ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										41
									
								
								.github/workflows/all-ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,41 @@ | ||||
| name: Build and Test the Prog8 compiler | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|   workflow_dispatch: | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - name: Checkout code | ||||
|         uses: actions/checkout@v4 | ||||
|  | ||||
|       - name: build and install recent 64tass | ||||
|         run: | | ||||
|           sudo apt-get install -y make build-essential | ||||
|           git clone --depth=1 https://github.com/irmen/64tass | ||||
|           cd 64tass | ||||
|           make -j4 | ||||
|           sudo make install           | ||||
|  | ||||
|       - name: Set up JDK 11 | ||||
|         uses: actions/setup-java@v4 | ||||
|         with: | ||||
|           java-version: 11 | ||||
|           distribution: temurin | ||||
|  | ||||
|       - name: Build and test with Gradle | ||||
|         run: | | ||||
|           ./gradlew build shadowJar --no-daemon | ||||
|           sha256sum -b compiler/build/libs/*-all.jar > compiler/build/libs/hash.txt | ||||
|          | ||||
|       - name: Create compiler shadowJar artifact | ||||
|         uses: actions/upload-artifact@v4 | ||||
|         with: | ||||
|           name: prog8-compiler-jar-zipped | ||||
|           path: | | ||||
|             compiler/build/libs/*-all.jar | ||||
|             compiler/build/libs/hash.txt | ||||
|  | ||||
							
								
								
									
										12
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										12
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -1,26 +1,34 @@ | ||||
| .idea/workspace.xml | ||||
| .idea/discord.xml | ||||
| .idea/developer-tools.xml | ||||
| .idea/usage.statistics.xml | ||||
| .idea/shelf/ | ||||
| build/ | ||||
| dist/ | ||||
| output/ | ||||
| out/ | ||||
| out-new/ | ||||
| out-old/ | ||||
| .*cache/ | ||||
| *.directory | ||||
| *.prg | ||||
| *.bin | ||||
| *.p8ir | ||||
| *.labels.txt | ||||
| *.vm.txt | ||||
| *.vice-mon-list | ||||
| docs/build | ||||
| out/ | ||||
| parser/**/*.interp | ||||
| parser/**/*.tokens | ||||
| parser/**/*.java | ||||
| compiler/src/prog8/buildversion/* | ||||
| *.py[cod] | ||||
| *.egg | ||||
| *.egg-info | ||||
| .eggs/ | ||||
| /MANIFEST | ||||
| .tox/ | ||||
| .kotlin/ | ||||
| __pycache__/ | ||||
| parser.out | ||||
| parsetab.py | ||||
| @@ -29,6 +37,6 @@ parsetab.py | ||||
| compiler/lib/ | ||||
|  | ||||
| .gradle | ||||
| /prog8compiler.jar | ||||
| sd*.img | ||||
| *.d64 | ||||
|  | ||||
|   | ||||
							
								
								
									
										4
									
								
								.idea/compiler.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										4
									
								
								.idea/compiler.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,7 +1,7 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project version="4"> | ||||
|   <component name="CompilerConfiguration"> | ||||
|     <option name="BUILD_PROCESS_HEAP_SIZE" value="1200" /> | ||||
|     <option name="BUILD_PROCESS_HEAP_SIZE" value="3000" /> | ||||
|     <bytecodeTargetLevel target="11" /> | ||||
|   </component> | ||||
| </project> | ||||
| </project> | ||||
							
								
								
									
										1581
									
								
								.idea/inspectionProfiles/Project_Default.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1581
									
								
								.idea/inspectionProfiles/Project_Default.xml
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										13
									
								
								.idea/kotlinc.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										13
									
								
								.idea/kotlinc.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,6 +1,19 @@ | ||||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project version="4"> | ||||
|   <component name="Kotlin2JsCompilerArguments"> | ||||
|     <option name="moduleKind" value="plain" /> | ||||
|   </component> | ||||
|   <component name="Kotlin2JvmCompilerArguments"> | ||||
|     <option name="jvmTarget" value="11" /> | ||||
|   </component> | ||||
|   <component name="KotlinCommonCompilerArguments"> | ||||
|     <option name="apiVersion" value="2.2" /> | ||||
|     <option name="languageVersion" value="2.2" /> | ||||
|   </component> | ||||
|   <component name="KotlinCompilerSettings"> | ||||
|     <option name="additionalArguments" value="-Xwhen-guards -jvm-default=no-compatibility" /> | ||||
|   </component> | ||||
|   <component name="KotlinJpsPluginSettings"> | ||||
|     <option name="version" value="2.2.0" /> | ||||
|   </component> | ||||
| </project> | ||||
							
								
								
									
										28
									
								
								.idea/libraries/KotlinJavaRuntime.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										28
									
								
								.idea/libraries/KotlinJavaRuntime.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,19 +1,23 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="KotlinJavaRuntime"> | ||||
|   <library name="KotlinJavaRuntime" type="repository"> | ||||
|     <properties maven-id="org.jetbrains.kotlin:kotlin-stdlib-jdk8:2.2.20" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-stdlib.jar!/" /> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-reflect.jar!/" /> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-test.jar!/" /> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-stdlib-jdk7.jar!/" /> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-stdlib-jdk8.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/2.2.20/kotlin-stdlib-jdk8-2.2.20.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/2.2.20/kotlin-stdlib-2.2.20.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/2.2.20/kotlin-stdlib-jdk7-2.2.20.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <JAVADOC> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/2.2.20/kotlin-stdlib-jdk8-2.2.20-javadoc.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/2.2.20/kotlin-stdlib-2.2.20-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/2.2.20/kotlin-stdlib-jdk7-2.2.20-javadoc.jar!/" /> | ||||
|     </JAVADOC> | ||||
|     <SOURCES> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-stdlib-sources.jar!/" /> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-reflect-sources.jar!/" /> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-test-sources.jar!/" /> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-stdlib-jdk7-sources.jar!/" /> | ||||
|       <root url="jar://$KOTLIN_BUNDLED$/lib/kotlin-stdlib-jdk8-sources.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/2.2.20/kotlin-stdlib-jdk8-2.2.20-sources.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/2.2.20/kotlin-stdlib-2.2.20-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/2.2.20/kotlin-stdlib-jdk7-2.2.20-sources.jar!/" /> | ||||
|     </SOURCES> | ||||
|   </library> | ||||
| </component> | ||||
							
								
								
									
										11
									
								
								.idea/libraries/antlr_antlr4.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										11
									
								
								.idea/libraries/antlr_antlr4.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,17 +1,16 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="antlr.antlr4" type="repository"> | ||||
|     <properties maven-id="org.antlr:antlr4:4.9.2"> | ||||
|     <properties maven-id="org.antlr:antlr4:4.13.2"> | ||||
|       <exclude> | ||||
|         <dependency maven-id="com.ibm.icu:icu4j" /> | ||||
|       </exclude> | ||||
|     </properties> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/antlr/antlr4/4.9.2/antlr4-4.9.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/antlr/antlr4-runtime/4.9.2/antlr4-runtime-4.9.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/antlr/antlr-runtime/3.5.2/antlr-runtime-3.5.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/antlr/ST4/4.3/ST4-4.3.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.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/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/glassfish/javax.json/1.0.4/javax.json-1.0.4.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   | ||||
							
								
								
									
										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.24.0" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/eclipse/lsp4j/org.eclipse.lsp4j/0.24.0/org.eclipse.lsp4j-0.24.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/eclipse/lsp4j/org.eclipse.lsp4j.jsonrpc/0.24.0/org.eclipse.lsp4j.jsonrpc-0.24.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/google/code/gson/gson/2.13.1/gson-2.13.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/google/errorprone/error_prone_annotations/2.38.0/error_prone_annotations-2.38.0.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   </library> | ||||
| </component> | ||||
							
								
								
									
										25
									
								
								.idea/libraries/github_hypfvieh_dbus_java.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										25
									
								
								.idea/libraries/github_hypfvieh_dbus_java.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,25 +0,0 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="github.hypfvieh.dbus.java" type="repository"> | ||||
|     <properties maven-id="com.github.hypfvieh:dbus-java:3.3.1" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/hypfvieh/dbus-java/3.3.1/dbus-java-3.3.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/jnr/jnr-unixsocket/0.38.6/jnr-unixsocket-0.38.6.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/jnr/jnr-ffi/2.2.2/jnr-ffi-2.2.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/jnr/jffi/1.3.1/jffi-1.3.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/jnr/jffi/1.3.1/jffi-1.3.1-native.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/ow2/asm/asm/9.1/asm-9.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/ow2/asm/asm-commons/9.1/asm-commons-9.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/ow2/asm/asm-analysis/9.1/asm-analysis-9.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/ow2/asm/asm-tree/9.1/asm-tree-9.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/ow2/asm/asm-util/9.1/asm-util-9.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/jnr/jnr-a64asm/1.0.0/jnr-a64asm-1.0.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/jnr/jnr-x86asm/1.0.2/jnr-x86asm-1.0.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/jnr/jnr-constants/0.10.1/jnr-constants-0.10.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/jnr/jnr-enxio/0.32.4/jnr-enxio-0.32.4.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/jnr/jnr-posix/3.1.5/jnr-posix-3.1.5.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/slf4j/slf4j-api/1.7.30/slf4j-api-1.7.30.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   </library> | ||||
| </component> | ||||
							
								
								
									
										10
									
								
								.idea/libraries/glassfish_javax_json.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										10
									
								
								.idea/libraries/glassfish_javax_json.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,10 +0,0 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="glassfish.javax.json" type="repository"> | ||||
|     <properties include-transitive-deps="false" maven-id="org.glassfish:javax.json:1.1.4" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/glassfish/javax.json/1.1.4/javax.json-1.1.4.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   </library> | ||||
| </component> | ||||
							
								
								
									
										22
									
								
								.idea/libraries/io_kotest_assertions_core_jvm.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										22
									
								
								.idea/libraries/io_kotest_assertions_core_jvm.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,22 +0,0 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="io.kotest.assertions.core.jvm" type="repository"> | ||||
|     <properties maven-id="io.kotest:kotest-assertions-core-jvm:4.6.3" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-core-jvm/4.6.3/kotest-assertions-core-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/1.5.0/kotlin-stdlib-jdk8-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.5.0/kotlin-stdlib-1.5.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.5.0/kotlin-stdlib-jdk7-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-shared-jvm/4.6.3/kotest-assertions-shared-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/github/java-diff-utils/java-diff-utils/4.9/java-diff-utils-4.9.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-jdk8/1.5.0/kotlinx-coroutines-jdk8-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-reflect/1.5.0/kotlin-reflect-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-core-jvm/1.5.0/kotlinx-coroutines-core-jvm-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-common/1.5.0/kotlin-stdlib-common-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-common-jvm/4.6.3/kotest-common-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-api-jvm/4.6.3/kotest-assertions-api-jvm-4.6.3.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   </library> | ||||
| </component> | ||||
							
								
								
									
										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> | ||||
							
								
								
									
										24
									
								
								.idea/libraries/io_kotest_property_jvm.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										24
									
								
								.idea/libraries/io_kotest_property_jvm.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,24 +0,0 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="io.kotest.property.jvm" type="repository"> | ||||
|     <properties maven-id="io.kotest:kotest-property-jvm:4.6.3" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-property-jvm/4.6.3/kotest-property-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/1.5.0/kotlin-stdlib-jdk8-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.5.0/kotlin-stdlib-1.5.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.5.0/kotlin-stdlib-jdk7-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-common-jvm/4.6.3/kotest-common-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-shared-jvm/4.6.3/kotest-assertions-shared-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-api-jvm/4.6.3/kotest-assertions-api-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-jdk8/1.5.0/kotlinx-coroutines-jdk8-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/github/java-diff-utils/java-diff-utils/4.9/java-diff-utils-4.9.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/github/mifmif/generex/1.0.2/generex-1.0.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/dk/brics/automaton/automaton/1.11-8/automaton-1.11-8.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-reflect/1.5.0/kotlin-reflect-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-core-jvm/1.5.0/kotlinx-coroutines-core-jvm-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-common/1.5.0/kotlin-stdlib-common-1.5.0.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   </library> | ||||
| </component> | ||||
							
								
								
									
										77
									
								
								.idea/libraries/io_kotest_runner_junit5_jvm.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										77
									
								
								.idea/libraries/io_kotest_runner_junit5_jvm.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,54 +1,39 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="io.kotest.runner.junit5.jvm" type="repository"> | ||||
|     <properties maven-id="io.kotest:kotest-runner-junit5-jvm:4.6.3" /> | ||||
|     <properties maven-id="io.kotest:kotest-runner-junit5-jvm:5.9.1" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-runner-junit5-jvm/4.6.3/kotest-runner-junit5-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-api-jvm/4.6.3/kotest-framework-api-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-shared-jvm/4.6.3/kotest-assertions-shared-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/github/java-diff-utils/java-diff-utils/4.9/java-diff-utils-4.9.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-common-jvm/4.6.3/kotest-common-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-engine-jvm/4.6.3/kotest-framework-engine-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/github/classgraph/classgraph/4.8.105/classgraph-4.8.105.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.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/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.8.0/kotlinx-coroutines-test-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$/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.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/colormath/1.2.0/colormath-1.2.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-script-util/1.5.0/kotlin-script-util-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/intellij/deps/trove4j/1.0.20181211/trove4j-1.0.20181211.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-daemon-client/1.5.0/kotlin-daemon-client-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-core/1.3.8/kotlinx-coroutines-core-1.3.8.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-scripting-jvm/1.5.0/kotlin-scripting-jvm-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-scripting-common/1.5.0/kotlin-scripting-common-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/opentest4j/opentest4j/1.2.0/opentest4j-1.2.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-discovery-jvm/4.6.3/kotest-framework-discovery-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-core-jvm/4.6.3/kotest-assertions-core-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-jdk8/1.5.0/kotlinx-coroutines-jdk8-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-assertions-api-jvm/4.6.3/kotest-assertions-api-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-extensions-jvm/4.6.3/kotest-extensions-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/commons-io/commons-io/2.6/commons-io-2.6.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/mockk/mockk/1.9.3/mockk-1.9.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/mockk/mockk-common/1.9.3/mockk-common-1.9.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/mockk/mockk-dsl/1.9.3/mockk-dsl-1.9.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/mockk/mockk-dsl-jvm/1.9.3/mockk-dsl-jvm-1.9.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/mockk/mockk-agent-jvm/1.9.3/mockk-agent-jvm-1.9.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/mockk/mockk-agent-api/1.9.3/mockk-agent-api-1.9.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/mockk/mockk-agent-common/1.9.3/mockk-agent-common-1.9.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/objenesis/objenesis/3.0.1/objenesis-3.0.1.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/net/bytebuddy/byte-buddy/1.9.10/byte-buddy-1.9.10.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/net/bytebuddy/byte-buddy-agent/1.9.10/byte-buddy-agent-1.9.10.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/io/kotest/kotest-framework-concurrency-jvm/4.6.3/kotest-framework-concurrency-jvm-4.6.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-coroutines-core-jvm/1.5.0/kotlinx-coroutines-core-jvm-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-common/1.5.0/kotlin-stdlib-common-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-engine/1.6.2/junit-platform-engine-1.6.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/apiguardian/apiguardian-api/1.1.0/apiguardian-api-1.1.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-commons/1.6.2/junit-platform-commons-1.6.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-suite-api/1.6.2/junit-platform-suite-api-1.6.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/junit/platform/junit-platform-launcher/1.6.2/junit-platform-launcher-1.6.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/junit/jupiter/junit-jupiter-api/5.6.2/junit-jupiter-api-5.6.2.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/1.5.0/kotlin-stdlib-jdk8-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.5.0/kotlin-stdlib-1.5.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.5.0/kotlin-stdlib-jdk7-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-script-runtime/1.5.0/kotlin-script-runtime-1.5.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-reflect/1.5.0/kotlin-reflect-1.5.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.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-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-agent/1.10.9/byte-buddy-agent-1.10.9.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.9.1/kotest-assertions-core-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/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.9.1/kotest-extensions-jvm-5.9.1.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.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/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/apiguardian/apiguardian-api/1.1.2/apiguardian-api-1.1.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/jupiter/junit-jupiter-api/5.8.2/junit-jupiter-api-5.8.2.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-reflect/1.9.23/kotlin-reflect-1.9.23.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   | ||||
							
								
								
									
										4
									
								
								.idea/libraries/jetbrains_kotlinx_cli_jvm.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										4
									
								
								.idea/libraries/jetbrains_kotlinx_cli_jvm.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,8 +1,8 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="jetbrains.kotlinx.cli.jvm" type="repository"> | ||||
|     <properties include-transitive-deps="false" maven-id="org.jetbrains.kotlinx:kotlinx-cli-jvm:0.3.3" /> | ||||
|     <properties include-transitive-deps="false" maven-id="org.jetbrains.kotlinx:kotlinx-cli-jvm:0.3.6" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-cli-jvm/0.3.3/kotlinx-cli-jvm-0.3.3.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlinx/kotlinx-cli-jvm/0.3.6/kotlinx-cli-jvm-0.3.6.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   | ||||
							
								
								
									
										21
									
								
								.idea/libraries/michael_bull_kotlin_result_jvm.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										21
									
								
								.idea/libraries/michael_bull_kotlin_result_jvm.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,15 +1,20 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="michael.bull.kotlin.result.jvm" type="repository"> | ||||
|     <properties maven-id="com.michael-bull.kotlin-result:kotlin-result-jvm:1.1.12" /> | ||||
|     <properties maven-id="com.michael-bull.kotlin-result:kotlin-result-jvm:2.1.0" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/michael-bull/kotlin-result/kotlin-result-jvm/1.1.12/kotlin-result-jvm-1.1.12.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-jdk8/1.5.10/kotlin-stdlib-jdk8-1.5.10.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/1.5.10/kotlin-stdlib-1.5.10.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/michael-bull/kotlin-result/kotlin-result-jvm/2.1.0/kotlin-result-jvm-2.1.0.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/2.2.0/kotlin-stdlib-2.2.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.5.10/kotlin-stdlib-jdk7-1.5.10.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib-common/1.5.10/kotlin-stdlib-common-1.5.10.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|     <JAVADOC> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/com/michael-bull/kotlin-result/kotlin-result-jvm/2.1.0/kotlin-result-jvm-2.1.0-javadoc.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/2.2.0/kotlin-stdlib-2.2.0-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.1.0/kotlin-result-jvm-2.1.0-sources.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/kotlin/kotlin-stdlib/2.2.0/kotlin-stdlib-2.2.0-sources.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/jetbrains/annotations/13.0/annotations-13.0-sources.jar!/" /> | ||||
|     </SOURCES> | ||||
|   </library> | ||||
| </component> | ||||
							
								
								
									
										11
									
								
								.idea/libraries/slf4j_simple.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										11
									
								
								.idea/libraries/slf4j_simple.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,11 +0,0 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="slf4j.simple" type="repository"> | ||||
|     <properties maven-id="org.slf4j:slf4j-simple:1.7.30" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/slf4j/slf4j-simple/1.7.30/slf4j-simple-1.7.30.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/slf4j/slf4j-api/1.7.30/slf4j-api-1.7.30.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   </library> | ||||
| </component> | ||||
							
								
								
									
										13
									
								
								.idea/libraries/takes.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										13
									
								
								.idea/libraries/takes.xml
									
									
									
										generated
									
									
									
								
							| @@ -1,13 +0,0 @@ | ||||
| <component name="libraryTable"> | ||||
|   <library name="takes" type="repository"> | ||||
|     <properties maven-id="org.takes:takes:1.19" /> | ||||
|     <CLASSES> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/takes/takes/1.19/takes-1.19.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/cactoos/cactoos/0.42/cactoos-0.42.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/apache/commons/commons-text/1.4/commons-text-1.4.jar!/" /> | ||||
|       <root url="jar://$MAVEN_REPOSITORY$/org/apache/commons/commons-lang3/3.7/commons-lang3-3.7.jar!/" /> | ||||
|     </CLASSES> | ||||
|     <JAVADOC /> | ||||
|     <SOURCES /> | ||||
|   </library> | ||||
| </component> | ||||
							
								
								
									
										13
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										13
									
								
								.idea/misc.xml
									
									
									
										generated
									
									
									
								
							| @@ -4,22 +4,29 @@ | ||||
|     <option name="perGrammarGenerationSettings"> | ||||
|       <list> | ||||
|         <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="outputDir" value="$PROJECT_DIR$/parser/src/prog8/parser" /> | ||||
|           <option name="libDir" value="" /> | ||||
|           <option name="encoding" value="" /> | ||||
|           <option name="pkg" value="" /> | ||||
|           <option name="language" value="" /> | ||||
|           <option name="language" value="Java" /> | ||||
|           <option name="generateListener" value="false" /> | ||||
|           <option name="generateVisitor" value="true" /> | ||||
|         </PerGrammarGenerationSettings> | ||||
|       </list> | ||||
|     </option> | ||||
|   </component> | ||||
|   <component name="Black"> | ||||
|     <option name="sdkName" value="Python 3.11" /> | ||||
|   </component> | ||||
|   <component name="FrameworkDetectionExcludesConfiguration"> | ||||
|     <type id="Python" /> | ||||
|   </component> | ||||
|   <component name="ProjectRootManager" version="2" languageLevel="JDK_11" project-jdk-name="11" project-jdk-type="JavaSDK"> | ||||
|   <component name="ProjectRootManager" version="2" languageLevel="JDK_21" default="true" project-jdk-name="openjdk-21" project-jdk-type="JavaSDK"> | ||||
|     <output url="file://$PROJECT_DIR$/out" /> | ||||
|   </component> | ||||
|   <component name="PythonCompatibilityInspectionAdvertiser"> | ||||
|     <option name="version" value="3" /> | ||||
|   </component> | ||||
| </project> | ||||
							
								
								
									
										15
									
								
								.idea/modules.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										15
									
								
								.idea/modules.xml
									
									
									
										generated
									
									
									
								
							| @@ -2,16 +2,23 @@ | ||||
| <project version="4"> | ||||
|   <component name="ProjectModuleManager"> | ||||
|     <modules> | ||||
|       <module fileurl="file://$PROJECT_DIR$/codeGeneration/codeGeneration.iml" filepath="$PROJECT_DIR$/codeGeneration/codeGeneration.iml" /> | ||||
|       <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$/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$/codeGenIntermediate/codeGenIntermediate.iml" filepath="$PROJECT_DIR$/codeGenIntermediate/codeGenIntermediate.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/codeGenVirtual/codeGenVirtual.iml" filepath="$PROJECT_DIR$/codeGenVirtual/codeGenVirtual.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/codeOptimizers/codeOptimizers.iml" filepath="$PROJECT_DIR$/codeOptimizers/codeOptimizers.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/compiler/compiler.iml" filepath="$PROJECT_DIR$/compiler/compiler.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/compilerAst/compilerAst.iml" filepath="$PROJECT_DIR$/compilerAst/compilerAst.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/compilerInterfaces/compilerInterfaces.iml" filepath="$PROJECT_DIR$/compilerInterfaces/compilerInterfaces.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/dbusCompilerService/dbusCompilerService.iml" filepath="$PROJECT_DIR$/dbusCompilerService/dbusCompilerService.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/docs/docs.iml" filepath="$PROJECT_DIR$/docs/docs.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/examples/examples.iml" filepath="$PROJECT_DIR$/examples/examples.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/httpCompilerService/httpCompilerService.iml" filepath="$PROJECT_DIR$/httpCompilerService/httpCompilerService.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$/.idea/modules/prog8.iml" filepath="$PROJECT_DIR$/.idea/modules/prog8.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/simpleAst/simpleAst.iml" filepath="$PROJECT_DIR$/simpleAst/simpleAst.iml" /> | ||||
|       <module fileurl="file://$PROJECT_DIR$/virtualmachine/virtualmachine.iml" filepath="$PROJECT_DIR$/virtualmachine/virtualmachine.iml" /> | ||||
|     </modules> | ||||
|   </component> | ||||
| </project> | ||||
							
								
								
									
										2
									
								
								.idea/vcs.xml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										2
									
								
								.idea/vcs.xml
									
									
									
										generated
									
									
									
								
							| @@ -3,4 +3,4 @@ | ||||
|   <component name="VcsDirectoryMappings"> | ||||
|     <mapping directory="$PROJECT_DIR$" vcs="Git" /> | ||||
|   </component> | ||||
| </project> | ||||
| </project> | ||||
| @@ -7,23 +7,21 @@ version: 2 | ||||
|  | ||||
| # Set the version of Python and other tools you might need | ||||
| build: | ||||
|   os: ubuntu-20.04 | ||||
|   os: ubuntu-22.04 | ||||
|   tools: | ||||
|     python: "3.9" | ||||
|     # You can also specify other tool versions: | ||||
|     # nodejs: "16" | ||||
|     # rust: "1.55" | ||||
|     # golang: "1.17" | ||||
|  | ||||
| # Build documentation in the docs/ directory with Sphinx | ||||
| sphinx: | ||||
|   configuration: docs/source/conf.py | ||||
|  | ||||
| # If using Sphinx, optionally build your docs in additional formats such as PDF | ||||
| formats: | ||||
|   - pdf | ||||
|     python: "3.12" | ||||
|  | ||||
| # Optionally declare the Python requirements required to build your docs | ||||
| python: | ||||
|   install: | ||||
|     - requirements: docs/requirements.txt | ||||
|  | ||||
| # Build documentation in the docs/ directory with Sphinx | ||||
| sphinx: | ||||
|   configuration: docs/source/conf.py | ||||
|   fail_on_warning: true | ||||
|  | ||||
| # If using Sphinx, optionally build your docs in additional formats such as PDF | ||||
| formats: | ||||
|   - pdf | ||||
|   - epub | ||||
|   | ||||
| @@ -1,32 +0,0 @@ | ||||
| #### Just a few remarks upfront: | ||||
| * There is the (gradle/IDEA) module `parser`: that's the parser generated by ANTLR4, in Java. The only file to be edited here is the grammar, `prog8.g4`. | ||||
| * Then we have the module `compilerAst` - in Kotlin - which uses `parser` and adds AST nodes. Here we put our additions to the generated thing, *including any tests of the parsing stage*. | ||||
|   - the name is a bit misleading, as this module isn't (or, resp. shouldn't be; see below) about *compiling*, only the parsing stage | ||||
|   - also, the tree that comes out isn't much of an *abstraction*, but rather still more or less a parse tree (this might very well change). | ||||
|   - **However, let's not *yet* rename the module.** We'll find a good name during refactoring. | ||||
|  | ||||
| #### Problems with `compilerAst`: | ||||
| * `ModuleImporter.kt`, doing (Prog8-) module resolution. That's not the parser's job. | ||||
| * `ParsingFailedError` (in `ModuleParsing.kt`): this exception (it is actually *not* a `java.lang.Error`...) is thrown in a number of places, where other exceptions would make more sense. For example: not finding a file should just yield a `NoSuchFileException`, not this one. The other problem with it is that it does not provide any additional information about the source of parsing error, in particular a `Position`. | ||||
| * During parsing, character literals are turned into UBYTEs (since there is no basic type e.g. CHAR). That's bad because it depends on a specific character encoding (`IStringEncoding` in `compilerAst/src/prog8/ast/AstToplevel.kt`) of/for some target platform. Note that *strings* are indeed encoded later, in the `compiler` module. | ||||
| * The same argument applies to `IMemSizer`, and - not entirely sure about that - `IBuiltinFunctions`. | ||||
|  | ||||
| #### Steps to take, in conceptual (!) order: | ||||
| 1. introduce an abstraction `SourceCode` that encapsulates the origin and actual loading of Prog8 source code | ||||
|    - from the local file system (use case: user programs) | ||||
|    - from resources (prog8lib) | ||||
|    - from plain strings (for testing) | ||||
| 2. add subclass `ParseError : ParsingFailedError` which adds information about the *source of parsing error* (`SourceCode` and `Position`). We cannot just replace `ParsingFailedError`  right away because it is so widely used (even in the `compiler` module). Therefore we'll just subclass for the time being, add more and more tests requiring the new one to be thrown (or, resp., NOT to be thrown), and gradually transition. | ||||
| 3. introduce a minimal interface to the outside, input: `SourceCode`, output: a tree with a `Module` node as the root | ||||
|    - this will be the Kotlin singleton `Prog8Parser` with the main method `parseModule` | ||||
|    - plus, optionally, method's for registering/unregistering a listener with the parser | ||||
|    - the *only* exception ever thrown / reported to listeners (TBD) will be `ParseError` | ||||
|    - anything related to the lexer, error strategies, character/token streams is hidden from the outside | ||||
|    - to make a clear distinction between the *generated* parser (and lexer) vs. `Prog8Parser`, and to discourage directly using the generated stuff, we'll rename the existing `prog8Parser`/`prog8Lexer` to `Prog8ANTLRParser` and `Prog8ANTLRLexer` and move them to package `prog8.parser.generated` | ||||
| 4. introduce AST node `CharLiteral` and keep them until after identifier resolution and type checking; insert there an AST transformation step that turns them in UBYTE constants (literals) | ||||
| 5. remove uses of `IStringEncoding` from module `compilerAst` - none should be necessary anymore | ||||
| 6. move `IStringEncoding` to module `compiler` | ||||
| 7. same with `ModuleImporter`, then rewrite that (addressing #46) | ||||
| 8. refactor AST nodes and grammar: less generated parse tree nodes (`XyzContext`), less intermediary stuff (private classes in `Antrl2Kotlin.kt`), more compact code. Also: nicer names such as simply `StringLiteral` instead of `StringLiteralValue` | ||||
| 9. re-think `IStringEncoding` to address #38 | ||||
|  | ||||
							
								
								
									
										7
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						
									
										7
									
								
								LICENSE
									
									
									
									
									
								
							| @@ -1,6 +1,9 @@ | ||||
|  | ||||
| This sofware license is for Prog8 the compiler + associated libraries. | ||||
| The software generated by running the compiler is excluded from this. | ||||
| This sofware license is for Prog8 the compiler + associated library files. | ||||
|  | ||||
| Exception: All output files generated by the compiler (intermediary files | ||||
| and compiled binary programs) are excluded from this; you can do with those | ||||
| whatever you want. | ||||
|  | ||||
|  | ||||
|  | ||||
|   | ||||
							
								
								
									
										12
									
								
								Makefile
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								Makefile
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | ||||
| # super simple Makefile to lauch the main gradle targets to build and/or test the prog8 compiler | ||||
|  | ||||
| .PHONY: all test | ||||
|  | ||||
| all: | ||||
| 	gradle installdist installshadowdist | ||||
| 	@echo "compiler launch script can be found here: compiler/build/install/compiler-shadow/bin/prog8c" | ||||
|  | ||||
| test: | ||||
| 	gradle build | ||||
| 	@echo "compiler launch script can be found here: compiler/build/install/compiler-shadow/bin/prog8c" | ||||
|  | ||||
							
								
								
									
										97
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										97
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,3 +1,7 @@ | ||||
| [](https://ko-fi.com/H2H6S0FFF) | ||||
|  | ||||
| PayPal: [](https://paypal.me/irmendejong) | ||||
|  | ||||
| [](https://prog8.readthedocs.io/) | ||||
|  | ||||
| Prog8 - Structured Programming Language for 8-bit 6502/65c02 microprocessors | ||||
| @@ -9,41 +13,76 @@ This is a structured programming language for the 8-bit 6502/6510/65c02 micropro | ||||
| as used in many home computers from that era. It is a medium to low level programming language, | ||||
| which aims to provide many conveniences over raw assembly code (even when using a macro assembler). | ||||
|  | ||||
| **Want to buy me a coffee or a pizza perhaps?** | ||||
|  | ||||
| This project was created over the last couple of years by dedicating thousands of hours of my free time to it, to make it the best I possibly can. | ||||
| If you like Prog8, and think it's worth a nice cup of hot coffee or a delicious pizza,  | ||||
| you can help me out a little bit over at [ko-fi.com/irmen](https://ko-fi.com/irmen) or [PayPal](https://paypal.me/irmendejong) | ||||
|  | ||||
|  | ||||
| Documentation | ||||
| ------------- | ||||
| Full documentation (syntax reference, how to use the language and the compiler, etc.) can be found at: | ||||
| https://prog8.readthedocs.io/ | ||||
|  | ||||
| How to get it/build it | ||||
| ---------------------- | ||||
|  | ||||
| - Download the latest [official release](https://github.com/irmen/prog8/releases) from github. | ||||
| - 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). | ||||
|   - 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: | ||||
|   - Arch (via AUR): [`prog8`](https://aur.archlinux.org/packages/prog8) | ||||
|  | ||||
| Community | ||||
| --------- | ||||
| Most of the development on Prog8 and the use of it is currently centered around  | ||||
| the [Commander X16](https://www.commanderx16.com/) retro computer. Their [discord server](https://discord.gg/nS2PqEC) contains a small channel | ||||
| dedicated to Prog8. Other than that, use the issue tracker on github. | ||||
|  | ||||
|  | ||||
| Software license | ||||
| ---------------- | ||||
| GNU GPL 3.0, see file LICENSE | ||||
| GNU GPL 3.0 (see file LICENSE), with exception for generated code: | ||||
|  | ||||
| - prog8 (the compiler + libraries) is licensed under GNU GPL 3.0 | ||||
| - *exception:* the resulting files created by running the compiler are free to use in whatever way desired. | ||||
| - The compiler and its libraries are free to use according to the terms of the GNU GPL 3.0 | ||||
| - *exception:* the resulting files (intermediate source codes and resulting binary program) created by the compiler | ||||
|   are excluded from the GPL and are free to use in whatever way desired, commercially or not. | ||||
|  | ||||
|  | ||||
| What does Prog8 provide? | ||||
| ------------------------ | ||||
|  | ||||
| - reduction of source code length over raw assembly | ||||
| - fast execution speed due to compilation to native assembly code. It's possible to write certain raster interrupt 'demoscene' effects purely in Prog8. | ||||
| - modularity, symbol scoping, subroutines | ||||
| - all advantages of a higher level language over having to write assembly code manually | ||||
| - programs run very fast because it's compiled to native machine code | ||||
| - 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) | ||||
| - floating point math is supported if the target system provides floating point library routines (C64 and Cx16 both do) | ||||
| - Structs and typed pointers | ||||
| - floating point math is supported on certain targets | ||||
| - access to most Kernal ROM routines as external subroutine definitions you can call normally | ||||
| - tight control over Zeropage usage | ||||
| - programs can be run multiple times without reloading because of automatic variable (re)initializations. | ||||
| - programs can be configured to execute in ROM | ||||
| - 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 | ||||
| - subroutines with input parameters and result values | ||||
| - high-level program optimizations | ||||
| - small program boilerplate/compilersupport overhead | ||||
| - programs can be run multiple times without reloading because of automatic variable (re)initializations. | ||||
| - conditional branches | ||||
| - 'when' statement to provide a concise jump table alternative to if/elseif chains | ||||
| - many built-in functions such as ``sin``, ``cos``, ``rnd``, ``abs``, ``min``, ``max``, ``sqrt``, ``msb``, ``rol``, ``ror``, ``swap``, ``sort`` and ``reverse`` | ||||
| - conditional branches that map 1:1 to cpu status flags | ||||
| - ``when`` statement to provide a concise jump table alternative to if/elseif chains | ||||
| - ``on .. goto`` statement for fast jump tables | ||||
| - ``in`` expression for concise and efficient multi-value/containment check  | ||||
| - ``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   | ||||
| - convenience abstractions for low level aspects such as ZeroPage handling, program startup, explicit memory addresses | ||||
| - subroutines can return more than one result value | ||||
| - 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. | ||||
| - encode strings and characters into petscii or screencodes as desired (C64/Cx16) | ||||
| - 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 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:* | ||||
|  | ||||
| @@ -52,11 +91,14 @@ What does Prog8 provide? | ||||
| - breakpoints, that let the Vice emulator drop into the monitor if execution hits them | ||||
| - source code labels automatically loaded in Vice emulator so it can show them in disassembly | ||||
|  | ||||
| *Two 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!): | ||||
|  | ||||
| - "c64": Commodore-64  (6510 CPU = almost a 6502) | ||||
| - "cx16": [CommanderX16](https://www.commanderx16.com)  (65c02 CPU) | ||||
| - If you only use standard kernal and prog8 library routines, it is possible to compile the *exact same program* for both machines (just change the compiler target flag)! | ||||
| - "c64": Commodore-64  (6502 like CPU) | ||||
| - "c128": Commodore-128  (6502 like CPU - the Z80 cpu mode is not supported) | ||||
| - "pet32": Commodore PET (limited support) | ||||
| - via external configurable targets: Atari 800 XL, Neo6502, NES, C64 OS, Foenix F256, ... | ||||
| - 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) | ||||
|  | ||||
|  | ||||
|  | ||||
| @@ -73,7 +115,12 @@ IntelliJ IDEA with the Kotlin plugin). | ||||
|  | ||||
| It's handy to have an emulator (or a real machine perhaps!) to run the programs on. The compiler assumes the presence | ||||
| of the [Vice emulator](http://vice-emu.sourceforge.net/)  for the C64 target, | ||||
| and the [x16emu emulator](https://github.com/commanderx16/x16-emulator) for the CommanderX16 target. | ||||
| and a recent emulator version (R42 or newer) for the CommanderX16, such as [x16emu](https://cx16forum.com/forum/viewforum.php?f=30)  | ||||
| (preferred, this is the official emulator. If required, source code is [here](https://github.com/X16Community/x16-emulator/)). | ||||
| There is also [Box16](https://github.com/indigodarkwolf/box16) which has powerful debugging features. | ||||
|  | ||||
| **Syntax highlighting:** for a few different editors, syntax highlighting definition files are provided. | ||||
| Look in the [syntax-files](https://github.com/irmen/prog8/tree/master/syntax-files) directory in the github repository to find them. | ||||
|  | ||||
|  | ||||
| Example code | ||||
| @@ -83,14 +130,13 @@ This code calculates prime numbers using the Sieve of Eratosthenes algorithm:: | ||||
|  | ||||
|     %import textio | ||||
|     %zeropage basicsafe | ||||
|      | ||||
|  | ||||
|     main { | ||||
|      | ||||
|         ubyte[256] sieve | ||||
|         bool[256] sieve | ||||
|         ubyte candidate_prime = 2       ; is increased in the loop | ||||
|      | ||||
|         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") | ||||
|             ubyte amount=0 | ||||
|             repeat { | ||||
| @@ -126,9 +172,6 @@ This code calculates prime numbers using the Sieve of Eratosthenes algorithm:: | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| when compiled an ran on a C-64 you'll get: | ||||
|  | ||||
|  | ||||
|   | ||||
| @@ -10,7 +10,15 @@ | ||||
|     <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="github.hypfvieh.dbus.java" level="project" /> | ||||
|     <orderEntry type="library" name="slf4j.simple" 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] | ||||
|     } | ||||
| } | ||||
							
								
								
									
										234
									
								
								benchmark-program/b_btree.p8
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										234
									
								
								benchmark-program/b_btree.p8
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,234 @@ | ||||
| ; Binary Search Tree. | ||||
| ; It's a simple implementation for test/demonstration purposes of the pointer support; | ||||
| ; no balancing is done and memory is not freed when elements are removed. | ||||
|  | ||||
| %import textio | ||||
|  | ||||
| btree { | ||||
|  | ||||
|     sub benchmark(uword max_time) -> uword { | ||||
|         txt.nl() | ||||
|         cbm.SETTIM(0,0,0) | ||||
|         uword score | ||||
|         while cbm.RDTIM16() < max_time { | ||||
|             bench_operations() | ||||
|             txt.chrout('.') | ||||
|             score++ | ||||
|         } | ||||
|         txt.nl() | ||||
|         return score | ||||
|     } | ||||
|  | ||||
|     sub bench_operations() { | ||||
|         arena.freeall() | ||||
|         btree.root = 0 | ||||
|  | ||||
|         for cx16.r0 in [321, 719, 194, 550, 187, 203, 520, 562, 221, 676, 97, 852, 273, 326, 589, 606, 275, 794, 63, 716] | ||||
|             btree.add(cx16.r0) | ||||
|  | ||||
|         cx16.r0L = btree.size() | ||||
|         btree.process_tree_inorder() | ||||
|         btree.process_tree_preorder() | ||||
|  | ||||
|         void btree.contains(203) | ||||
|         void btree.contains(204) | ||||
|         void btree.contains(605) | ||||
|         void btree.contains(606) | ||||
|  | ||||
|         btree.remove(9999) | ||||
|         btree.remove(97) | ||||
|         btree.remove(187) | ||||
|         btree.remove(203) | ||||
|         btree.remove(275) | ||||
|         btree.remove(321) | ||||
|         btree.remove(520) | ||||
|         btree.remove(562) | ||||
|         btree.remove(606) | ||||
|         btree.remove(719) | ||||
|         btree.remove(794) | ||||
|  | ||||
|         cx16.r0L = btree.size() | ||||
|         btree.process_tree_inorder() | ||||
|         btree.process_tree_preorder() | ||||
|     } | ||||
|  | ||||
|     struct Node { | ||||
|         ^^Node left | ||||
|         ^^Node right | ||||
|         uword value | ||||
|     } | ||||
|  | ||||
|     ^^Node root = 0 | ||||
|  | ||||
|     sub add(uword value) { | ||||
|         ^^Node node = arena.alloc(sizeof(Node)) | ||||
|         node.value = value | ||||
|         node.left = node.right = 0 | ||||
|  | ||||
|         if root==0 | ||||
|             root=node | ||||
|         else { | ||||
|             ^^Node parent = root | ||||
|             repeat { | ||||
|                 if parent.value >= value { | ||||
|                     if parent.left!=0 | ||||
|                         parent = parent.left | ||||
|                     else { | ||||
|                         parent.left = node | ||||
|                         return | ||||
|                     } | ||||
|                 } else { | ||||
|                     if parent.right!=0 | ||||
|                         parent = parent.right | ||||
|                     else { | ||||
|                         parent.right = node | ||||
|                         return | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     sub contains(uword value) -> bool { | ||||
|         ^^Node r = root | ||||
|         while r!=0 { | ||||
|             if r.value==value | ||||
|                 return true | ||||
|             if r.value>value | ||||
|                 r = r.left | ||||
|             else | ||||
|                 r = r.right | ||||
|         } | ||||
|         return false | ||||
|     } | ||||
|  | ||||
|     sub size() -> ubyte { | ||||
|         ubyte count | ||||
|  | ||||
|         if root!=0 | ||||
|             count_node(root) | ||||
|  | ||||
|         return count | ||||
|  | ||||
|         sub count_node(^^Node r) { | ||||
|             count++ | ||||
|             if r.left!=0 { | ||||
|                 sys.pushw(r) | ||||
|                 count_node(r.left) | ||||
|                 r = sys.popw() | ||||
|             } | ||||
|             if r.right!=0 { | ||||
|                 sys.pushw(r) | ||||
|                 count_node(r.right) | ||||
|                 r = sys.popw() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     sub remove(uword value) { | ||||
|         ; note: we don't deallocate the memory from the node, for simplicity sake | ||||
|         ^^Node n = root | ||||
|         ^^Node parent = 0 | ||||
|         while n!=0 { | ||||
|             if n.value==value { | ||||
|                 if n.left==0 | ||||
|                     replacechild(parent, n, n.right) | ||||
|                 else if n.right==0 | ||||
|                     replacechild(parent, n, n.left) | ||||
|                 else { | ||||
|                     ; Both left & right subtrees are present. | ||||
|                     ; N = node to delete. | ||||
|                     ;    Find N's successor S. (N's right subtree's minimum element) | ||||
|                     ;    Attach N's left subtree to S.left (S doesn't have a left child) | ||||
|                     ;    Attach N's right subtree to Parent in place of N. | ||||
|                     ^^Node successor = find_successor(n) | ||||
|                     successor.left = n.left | ||||
|                     replacechild(parent, n, n.right) | ||||
|                 } | ||||
|                 return | ||||
|             } | ||||
|             parent = n | ||||
|             if n.value>value | ||||
|                 n = n.left | ||||
|             else | ||||
|                 n = n.right | ||||
|         } | ||||
|  | ||||
|         sub find_successor(^^Node p) -> ^^Node { | ||||
|             ^^Node succ = p | ||||
|             p = p.right | ||||
|             while p!=0 { | ||||
|                 succ = p | ||||
|                 p = p.left | ||||
|             } | ||||
|             return succ | ||||
|         } | ||||
|  | ||||
|         sub replacechild(^^Node p, ^^Node child, ^^Node newchild) { | ||||
|             if p.left==child | ||||
|                 p.left = newchild | ||||
|             else | ||||
|                 p.right = newchild | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     sub process_tree_inorder() { | ||||
|         if root!=0 | ||||
|             process_tree(root) | ||||
|  | ||||
|         sub process_tree(^^Node r) { | ||||
|             if r.left!=0 { | ||||
|                 sys.pushw(r) | ||||
|                 process_tree(r.left) | ||||
|                 r = sys.popw() | ||||
|             } | ||||
|             cx16.r0 = r.value | ||||
|             if r.right!=0 { | ||||
|                 sys.pushw(r) | ||||
|                 process_tree(r.right) | ||||
|                 r = sys.popw() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|  | ||||
|     sub process_tree_preorder() { | ||||
|         if root!=0 | ||||
|             process_tree(root,0) | ||||
|  | ||||
|         sub process_tree(^^Node r, ubyte depth) { | ||||
|             cx16.r0 = r.value | ||||
|             if r.left!=0 { | ||||
|                 sys.pushw(r) | ||||
|                 sys.push(depth) | ||||
|                 process_tree(r.left, depth+1) | ||||
|                 depth = sys.pop() | ||||
|                 r = sys.popw() | ||||
|             } | ||||
|             if r.right!=0 { | ||||
|                 sys.pushw(r) | ||||
|                 sys.push(depth) | ||||
|                 process_tree(r.right, depth+1) | ||||
|                 depth = sys.pop() | ||||
|                 r = sys.popw() | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| arena { | ||||
|     ; extremely trivial arena allocator (that never frees) | ||||
|     uword buffer = memory("arena", 2000, 0) | ||||
|     uword next = buffer | ||||
|  | ||||
|     sub alloc(ubyte size) -> uword { | ||||
|         defer next += size | ||||
|         return next | ||||
|     } | ||||
|  | ||||
|     sub freeall() { | ||||
|         next = buffer | ||||
|     } | ||||
| } | ||||
							
								
								
									
										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 | ||||
|             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 | ||||
|             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 | ||||
|     } | ||||
| } | ||||
							
								
								
									
										69
									
								
								benchmark-program/b_sprites.p8
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								benchmark-program/b_sprites.p8
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| %import sprites | ||||
| %import coroutines | ||||
| %import math | ||||
|  | ||||
|  | ||||
| animsprites { | ||||
|     uword num_iterations | ||||
|     ubyte[64] sx | ||||
|     ubyte[64] sy | ||||
|     ubyte[64] sc | ||||
|     ubyte[64] dx | ||||
|     ubyte[64] dy | ||||
|     uword maximum_duration | ||||
|  | ||||
|     sub benchmark(uword max_duration) -> uword { | ||||
|         maximum_duration = max_duration | ||||
|         math.rndseed(1122,9876) | ||||
|         cx16.set_screen_mode(3) | ||||
|         cx16.mouse_config2(1) | ||||
|         sprites.set_mousepointer_hand() | ||||
|         repeat 64 | ||||
|             void coroutines.add(animsprite, 0) | ||||
|         cx16.mouse_config2(0) | ||||
|  | ||||
|         cbm.SETTIM(0,0,0) | ||||
|         coroutines.run(supervisor) | ||||
|  | ||||
|         sprites.reset(0, 64) | ||||
|         return num_iterations | ||||
|     } | ||||
|  | ||||
|     sub supervisor() -> bool { | ||||
|         if cbm.RDTIM16() >= maximum_duration { | ||||
|             coroutines.killall() | ||||
|             return false | ||||
|         } | ||||
|         return true | ||||
|     } | ||||
|  | ||||
|     sub animsprite() { | ||||
|         num_iterations++ | ||||
|         ; set up the sprite | ||||
|         ubyte sprnum = coroutines.current() | ||||
|         cx16.r6L, cx16.r7 = sprites.get_data_ptr(0) | ||||
|         sprites.init(sprnum, cx16.r6L, cx16.r7, sprites.SIZE_16, sprites.SIZE_16, sprites.COLORS_256, 0) | ||||
|         sx[sprnum] = math.rnd() | ||||
|         sy[sprnum] = math.rnd() | ||||
|         sc[sprnum] = math.rnd() | ||||
|         dx[sprnum] = if math.rnd()&1 == 1  1 else 255 | ||||
|         dy[sprnum] = if math.rnd()&1 == 1  1 else 255 | ||||
|  | ||||
|         ; move the sprite around | ||||
|         while sc[sprnum]!=0 { | ||||
|             animate(sprnum) | ||||
|             void coroutines.yield() | ||||
|             sprnum = coroutines.current() | ||||
|         } | ||||
|  | ||||
|         sub animate(ubyte spr) { | ||||
|             defer sc[spr]-- | ||||
|             sprites.pos(spr, sx[spr], sy[spr]) | ||||
|             sx[spr] += dx[spr] | ||||
|             sy[spr] += dy[spr] | ||||
|         } | ||||
|  | ||||
|         ; end the task but replace it with a fresh animated sprite task | ||||
|         void coroutines.add(animsprite, 0) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										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++ | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										118
									
								
								benchmark-program/benchmark.p8
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										118
									
								
								benchmark-program/benchmark.p8
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,118 @@ | ||||
|  | ||||
| ;  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 | ||||
| %import b_sprites | ||||
| %import b_btree | ||||
|  | ||||
| %zeropage basicsafe | ||||
| %option no_sysinit | ||||
|  | ||||
|  | ||||
| main { | ||||
|  | ||||
|     str[20] benchmark_names | ||||
|     uword[20] benchmark_score | ||||
|  | ||||
|  | ||||
|     sub start() { | ||||
|         ubyte benchmark_number | ||||
|  | ||||
|         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("text-elite") | ||||
|         benchmark_score[benchmark_number]  = textelite.bench(120) | ||||
|         benchmark_number++ | ||||
|  | ||||
|         announce_benchmark("sprites-coroutines-defer") | ||||
|         benchmark_score[benchmark_number]  = animsprites.benchmark(300) | ||||
|         benchmark_number++ | ||||
|  | ||||
|         announce_benchmark("btree-struct-pointers") | ||||
|         benchmark_score[benchmark_number]  = btree.benchmark(200) | ||||
|         benchmark_number++ | ||||
|  | ||||
|         benchmark_names[benchmark_number] = 0 | ||||
|         benchmark_score[benchmark_number] = 0 | ||||
|  | ||||
|         cx16.set_screen_mode(3) | ||||
|         txt.uppercase() | ||||
|         txt.color2(1, 6) | ||||
|         uword total_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]) | ||||
|             total_score += benchmark_score[benchmark_number] | ||||
|             txt.nl() | ||||
|             benchmark_number++ | ||||
|         } until benchmark_names[benchmark_number]==0 | ||||
|  | ||||
|         txt.print("\n\ntotal score : ") | ||||
|         txt.print_uw(total_score) | ||||
|         txt.print(" (higher=better)\n") | ||||
|  | ||||
|         sub announce_benchmark(str name) { | ||||
|             benchmark_names[benchmark_number] = name | ||||
|             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() | ||||
|     } | ||||
| } | ||||
							
								
								
									
										38
									
								
								build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| import org.jetbrains.kotlin.gradle.dsl.JvmTarget | ||||
| import org.jetbrains.kotlin.gradle.dsl.JvmDefaultMode | ||||
| import org.jetbrains.kotlin.gradle.internal.config.LanguageFeature | ||||
| import org.jetbrains.kotlin.gradle.dsl.KotlinVersion | ||||
|  | ||||
|  | ||||
| plugins { | ||||
|     kotlin("jvm") version "2.2.20" | ||||
| } | ||||
|  | ||||
| allprojects { | ||||
|     apply(plugin="kotlin") | ||||
|  | ||||
|     repositories { | ||||
|         mavenLocal() | ||||
|         mavenCentral() | ||||
|     } | ||||
|  | ||||
|     kotlin { | ||||
|         compilerOptions { | ||||
|             freeCompilerArgs = listOf("-Xwhen-guards") | ||||
|             jvmTarget = JvmTarget.JVM_11 | ||||
|             jvmDefault = JvmDefaultMode.NO_COMPATIBILITY | ||||
|             // languageVersion.set(KotlinVersion.KOTLIN_2_3) | ||||
|         } | ||||
|         sourceSets.all { | ||||
|             languageSettings { | ||||
|                 // enable language features like so: | ||||
|                 // enableLanguageFeature(LanguageFeature.WhenGuards.name) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     java { | ||||
|         targetCompatibility = JavaVersion.VERSION_11 | ||||
|         sourceCompatibility = JavaVersion.VERSION_11 | ||||
|     } | ||||
| } | ||||
							
								
								
									
										22
									
								
								codeCore/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								codeCore/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| 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.1.0") | ||||
| } | ||||
|  | ||||
| sourceSets { | ||||
|     main { | ||||
|         java { | ||||
|             srcDir("${project.projectDir}/src") | ||||
|         } | ||||
|         resources { | ||||
|             srcDir("${project.projectDir}/res") | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| // note: there are no unit tests in this module! | ||||
| @@ -10,7 +10,5 @@ | ||||
|     <orderEntry type="sourceFolder" forTests="false" /> | ||||
|     <orderEntry type="library" name="KotlinJavaRuntime" level="project" /> | ||||
|     <orderEntry type="library" name="michael.bull.kotlin.result.jvm" level="project" /> | ||||
|     <orderEntry type="module" module-name="compilerInterfaces" /> | ||||
|     <orderEntry type="module" module-name="compilerAst" /> | ||||
|   </component> | ||||
| </module> | ||||
							
								
								
									
										32
									
								
								codeCore/src/prog8/code/Globals.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								codeCore/src/prog8/code/Globals.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| package prog8.code | ||||
|  | ||||
| import java.io.IOException | ||||
| import java.nio.file.Path | ||||
| import kotlin.io.path.absolute | ||||
|  | ||||
|  | ||||
| const val INTERNED_STRINGS_MODULENAME = "prog8_interned_strings" | ||||
|  | ||||
| val PROG8_CONTAINER_MODULES = arrayOf(INTERNED_STRINGS_MODULENAME)      // option to add more if needed one day | ||||
|  | ||||
| // all automatically generated labels everywhere need to have the same label name prefix: | ||||
| const val GENERATED_LABEL_PREFIX = "p8_label_gen_" | ||||
|  | ||||
|  | ||||
| /** | ||||
|  * Returns the absolute path of the given path, | ||||
|  * where links are replaced by the actual directories, | ||||
|  * and containing no redundant path elements. | ||||
|  * If the path doesn't refer to an existing directory or file on the file system, | ||||
|  * it is returned unchanged. | ||||
|  */ | ||||
| fun Path.sanitize(): Path { | ||||
|     return try { | ||||
|         this.toRealPath().normalize() | ||||
|     } catch (_: java.nio.file.NoSuchFileException) { | ||||
|         this.absolute().normalize() | ||||
|         //throw NoSuchFileException(this.toFile(), null, nx.reason).also { it.initCause(nx) } | ||||
|     } catch (iox: IOException) { | ||||
|         throw FileSystemException(this.toFile()).also { it.initCause(iox) } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										159
									
								
								codeCore/src/prog8/code/core/BuiltinFunctions.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										159
									
								
								codeCore/src/prog8/code/core/BuiltinFunctions.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,159 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| class ReturnConvention(val dt: BaseDataType?, val reg: RegisterOrPair?) | ||||
| class ParamConvention(val dt: BaseDataType, val reg: RegisterOrPair?, val variable: Boolean) | ||||
| class CallConvention(val params: List<ParamConvention>, val returns: ReturnConvention) { | ||||
|     override fun toString(): String { | ||||
|         val paramConvs =  params.mapIndexed { index, it -> | ||||
|             when { | ||||
|                 it.reg!=null -> "$index:${it.reg}" | ||||
|                 it.variable -> "$index:variable" | ||||
|                 else -> "$index:???" | ||||
|             } | ||||
|         } | ||||
|         val returnConv = | ||||
|             when { | ||||
|                 returns.reg == RegisterOrPair.FAC1 -> "floatFAC1" | ||||
|                 returns.reg != null -> returns.reg.toString() | ||||
|                 else -> "<no returnvalue>" | ||||
|             } | ||||
|         return "CallConvention[" + paramConvs.joinToString() + " ; returns: $returnConv]" | ||||
|     } | ||||
| } | ||||
|  | ||||
| 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) | ||||
|  | ||||
|  | ||||
| class FSignature(val pure: Boolean,      // does it have side effects? | ||||
|                  val returnType: BaseDataType?, | ||||
|                  vararg val parameters: FParam) { | ||||
|  | ||||
|     fun callConvention(actualParamTypes: List<BaseDataType>): CallConvention { | ||||
|         val returns: ReturnConvention = when (returnType) { | ||||
|             BaseDataType.UBYTE, BaseDataType.BYTE -> ReturnConvention(returnType, RegisterOrPair.A) | ||||
|             BaseDataType.UWORD, BaseDataType.WORD -> ReturnConvention(returnType, RegisterOrPair.AY) | ||||
|             BaseDataType.FLOAT -> ReturnConvention(returnType, RegisterOrPair.FAC1) | ||||
|             in IterableDatatypes -> ReturnConvention(returnType!!, RegisterOrPair.AY) | ||||
|             null -> ReturnConvention(null, null) | ||||
|             else -> { | ||||
|                 // return type depends on arg type | ||||
|                 when (val paramType = actualParamTypes.first()) { | ||||
|                     BaseDataType.UBYTE, BaseDataType.BYTE -> ReturnConvention(paramType, RegisterOrPair.A) | ||||
|                     BaseDataType.UWORD, BaseDataType.WORD -> ReturnConvention(paramType, RegisterOrPair.AY) | ||||
|                     BaseDataType.FLOAT -> ReturnConvention(paramType, RegisterOrPair.FAC1) | ||||
|                     in IterableDatatypes -> ReturnConvention(paramType, RegisterOrPair.AY) | ||||
|                     else -> ReturnConvention(paramType, null) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return when { | ||||
|             actualParamTypes.isEmpty() -> CallConvention(emptyList(), returns) | ||||
|             actualParamTypes.size==1 -> { | ||||
|                 // 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]) { | ||||
|                     BaseDataType.UBYTE, BaseDataType.BYTE -> ParamConvention(paramType, RegisterOrPair.A, false) | ||||
|                     BaseDataType.UWORD, BaseDataType.WORD -> 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 IterableDatatypes -> ParamConvention(paramType, RegisterOrPair.AY, false) | ||||
|                     else -> ParamConvention(paramType, null, false) | ||||
|                 } | ||||
|                 CallConvention(listOf(paramConv), returns) | ||||
|             } | ||||
|             actualParamTypes.size==2 && (actualParamTypes[0].isByte && 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 -> { | ||||
|                 // multiple parameters go via variables | ||||
|                 val paramConvs = actualParamTypes.map { ParamConvention(it, null, true) } | ||||
|                 CallConvention(paramConvs, returns) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| val BuiltinFunctions: Map<String, FSignature> = mapOf( | ||||
|     "setlsb"  to FSignature(false, null, FParam("variable", BaseDataType.WORD, BaseDataType.UWORD), FParam("value", BaseDataType.BYTE, BaseDataType.UBYTE)), | ||||
|     "setmsb"  to FSignature(false, null, FParam("variable", BaseDataType.WORD, BaseDataType.UWORD), FParam("value", BaseDataType.BYTE, BaseDataType.UBYTE)), | ||||
|     "rol"     to FSignature(false, null, FParam("item", BaseDataType.UBYTE, BaseDataType.UWORD)), | ||||
|     "ror"     to FSignature(false, null, FParam("item", BaseDataType.UBYTE, BaseDataType.UWORD)), | ||||
|     "rol2"    to FSignature(false, null, FParam("item", BaseDataType.UBYTE, BaseDataType.UWORD)), | ||||
|     "ror2"    to FSignature(false, null, FParam("item", BaseDataType.UBYTE, BaseDataType.UWORD)), | ||||
|     "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 | ||||
|     "prog8_lib_stringcompare"     to FSignature(true, BaseDataType.BYTE, FParam("str1", BaseDataType.STR), FParam("str2", BaseDataType.STR)), | ||||
|     "prog8_lib_square_byte"       to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.BYTE, BaseDataType.UBYTE)), | ||||
|     "prog8_lib_square_word"       to FSignature(true, BaseDataType.UWORD, FParam("value", BaseDataType.WORD, BaseDataType.UWORD)), | ||||
|     "prog8_lib_structalloc"       to FSignature(true, BaseDataType.UWORD), | ||||
|     "abs"           to FSignature(true, null, FParam("value", *NumericDatatypes)), | ||||
|     "abs__byte"     to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.BYTE)), | ||||
|     "abs__word"     to FSignature(true, BaseDataType.UWORD, FParam("value", BaseDataType.WORD)), | ||||
|     "abs__float"    to FSignature(true, BaseDataType.FLOAT, FParam("value", BaseDataType.FLOAT)), | ||||
|     "len"           to FSignature(true, BaseDataType.UWORD, FParam("values", *IterableDatatypes)), | ||||
|     "sizeof"        to FSignature(true, BaseDataType.UBYTE, FParam("object", *(BaseDataType.entries - BaseDataType.STRUCT_INSTANCE).toTypedArray())), | ||||
|     "offsetof"      to FSignature(true, BaseDataType.UBYTE, FParam("field", BaseDataType.UBYTE)), | ||||
|     "sgn"           to FSignature(true, BaseDataType.BYTE, FParam("value", *NumericDatatypes)), | ||||
|     "sqrt"          to FSignature(true, null, FParam("value", *NumericDatatypes)), | ||||
|     "sqrt__ubyte"   to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UBYTE)), | ||||
|     "sqrt__uword"   to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UWORD)), | ||||
|     "sqrt__float"   to FSignature(true, BaseDataType.FLOAT, FParam("value", BaseDataType.FLOAT)), | ||||
|     "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)), | ||||
|     "divmod__ubyte" to FSignature(false, null, FParam("dividend", BaseDataType.UBYTE), FParam("divisor", BaseDataType.UBYTE), FParam("quotient", BaseDataType.UBYTE), FParam("remainder", BaseDataType.UBYTE)), | ||||
|     "divmod__uword" to FSignature(false, null, FParam("dividend", BaseDataType.UWORD), FParam("divisor", BaseDataType.UWORD), FParam("quotient", BaseDataType.UWORD), FParam("remainder", BaseDataType.UWORD)), | ||||
|     "lsb"           to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)), | ||||
|     "lsw"           to FSignature(true, BaseDataType.UWORD, FParam("value", BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)), | ||||
|     "msb"           to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)), | ||||
|     "msw"           to FSignature(true, BaseDataType.UWORD, FParam("value", BaseDataType.UWORD, BaseDataType.WORD, BaseDataType.LONG)), | ||||
|     "mkword"        to FSignature(true, BaseDataType.UWORD, FParam("msb", BaseDataType.UBYTE), FParam("lsb", BaseDataType.UBYTE)), | ||||
|     "clamp"         to FSignature(true, null, FParam("value", BaseDataType.BYTE), FParam("minimum", BaseDataType.BYTE), FParam("maximum", BaseDataType.BYTE)), | ||||
|     "clamp__byte"   to FSignature(true, BaseDataType.BYTE, FParam("value", BaseDataType.BYTE), FParam("minimum", BaseDataType.BYTE), FParam("maximum", BaseDataType.BYTE)), | ||||
|     "clamp__ubyte"  to FSignature(true, BaseDataType.UBYTE, FParam("value", BaseDataType.UBYTE), FParam("minimum", BaseDataType.UBYTE), FParam("maximum", BaseDataType.UBYTE)), | ||||
|     "clamp__word"   to FSignature(true, BaseDataType.WORD, FParam("value", BaseDataType.WORD), FParam("minimum", BaseDataType.WORD), FParam("maximum", BaseDataType.WORD)), | ||||
|     "clamp__uword"  to FSignature(true, BaseDataType.UWORD, FParam("value", BaseDataType.UWORD), FParam("minimum", BaseDataType.UWORD), FParam("maximum", BaseDataType.UWORD)), | ||||
|     "min"           to FSignature(true, null, FParam("val1", BaseDataType.BYTE), FParam("val2", BaseDataType.BYTE)), | ||||
|     "min__byte"     to FSignature(true, BaseDataType.BYTE, FParam("val1", BaseDataType.BYTE), FParam("val2", BaseDataType.BYTE)), | ||||
|     "min__ubyte"    to FSignature(true, BaseDataType.UBYTE, FParam("val1", BaseDataType.UBYTE), FParam("val2", BaseDataType.UBYTE)), | ||||
|     "min__word"     to FSignature(true, BaseDataType.WORD, FParam("val1", BaseDataType.WORD), FParam("val2", BaseDataType.WORD)), | ||||
|     "min__uword"    to FSignature(true, BaseDataType.UWORD, FParam("val1", BaseDataType.UWORD), FParam("val2", BaseDataType.UWORD)), | ||||
|     "max"           to FSignature(true, null, FParam("val1", BaseDataType.BYTE), FParam("val2", BaseDataType.BYTE)), | ||||
|     "max__byte"     to FSignature(true, BaseDataType.BYTE, FParam("val1", BaseDataType.BYTE), FParam("val2", BaseDataType.BYTE)), | ||||
|     "max__ubyte"    to FSignature(true, BaseDataType.UBYTE, FParam("val1", BaseDataType.UBYTE), FParam("val2", BaseDataType.UBYTE)), | ||||
|     "max__word"     to FSignature(true, BaseDataType.WORD, FParam("val1", BaseDataType.WORD), FParam("val2", BaseDataType.WORD)), | ||||
|     "max__uword"    to FSignature(true, BaseDataType.UWORD, FParam("val1", BaseDataType.UWORD), FParam("val2", BaseDataType.UWORD)), | ||||
|     "peek"          to FSignature(true, BaseDataType.UBYTE, FParam("address", BaseDataType.UWORD)), | ||||
|     "peekbool"      to FSignature(true, BaseDataType.BOOL, FParam("address", BaseDataType.UWORD)), | ||||
|     "peekw"         to FSignature(true, BaseDataType.UWORD, FParam("address", BaseDataType.UWORD)), | ||||
|     "peekl"         to FSignature(true, BaseDataType.LONG, FParam("address", BaseDataType.UWORD)), | ||||
|     "peekf"         to FSignature(true, BaseDataType.FLOAT, FParam("address", BaseDataType.UWORD)), | ||||
|     "poke"          to FSignature(false, null, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.UBYTE)), | ||||
|     "pokebool"      to FSignature(false, null, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.BOOL)), | ||||
|     "pokebowl"      to FSignature(false, null, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.BOOL)), | ||||
|     "pokew"         to FSignature(false, null, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.UWORD)), | ||||
|     "pokel"         to FSignature(false, null, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.LONG)), | ||||
|     "pokef"         to FSignature(false, null, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.FLOAT)), | ||||
|     "pokemon"       to FSignature(false, BaseDataType.UBYTE, FParam("address", BaseDataType.UWORD), FParam("value", BaseDataType.UBYTE)), | ||||
|     "rsave"         to FSignature(false, null), | ||||
|     "rrestore"      to FSignature(false, null), | ||||
|     "memory"        to FSignature(true, BaseDataType.UWORD, FParam("name", BaseDataType.STR), FParam("size", BaseDataType.UWORD), FParam("alignment", BaseDataType.UWORD)), | ||||
|     "callfar"       to FSignature(false, BaseDataType.UWORD, FParam("bank", BaseDataType.UBYTE), FParam("address", BaseDataType.UWORD), FParam("arg", BaseDataType.UWORD)), | ||||
|     "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)), | ||||
|     "call"          to FSignature(false, BaseDataType.UWORD, FParam("address", BaseDataType.UWORD)), | ||||
| ) | ||||
|  | ||||
| val InplaceModifyingBuiltinFunctions = setOf( | ||||
|     "setlsb", "setmsb", | ||||
|     "rol", "ror", "rol2", "ror2", | ||||
|     "divmod", "divmod__ubyte", "divmod__uword" | ||||
| ) | ||||
							
								
								
									
										45
									
								
								codeCore/src/prog8/code/core/CompilationOptions.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								codeCore/src/prog8/code/core/CompilationOptions.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,45 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| import java.nio.file.Path | ||||
| import kotlin.io.path.Path | ||||
|  | ||||
|  | ||||
| class CompilationOptions(val output: OutputType, | ||||
|                          val launcher: CbmPrgLauncherType, | ||||
|                          val zeropage: ZeropageType, | ||||
|                          val zpReserved: List<UIntRange>, | ||||
|                          val zpAllowed: List<UIntRange>, | ||||
|                          val floats: Boolean, | ||||
|                          val noSysInit: Boolean, | ||||
|                          val romable: Boolean, | ||||
|                          val compTarget: ICompilationTarget, | ||||
|                          // these are set later, based on command line arguments or options in the source code: | ||||
|                          var loadAddress: UInt, | ||||
|                          var memtopAddress: UInt, | ||||
|                          var warnSymbolShadowing: Boolean = false, | ||||
|                          var optimize: Boolean = false, | ||||
|                          var asmQuiet: Boolean = false, | ||||
|                          var asmListfile: Boolean = false, | ||||
|                          var includeSourcelines: Boolean = false, | ||||
|                          var dumpVariables: Boolean = false, | ||||
|                          var dumpSymbols: Boolean = false, | ||||
|                          var experimentalCodegen: Boolean = false, | ||||
|                          var varsHighBank: Int? = null, | ||||
|                          var varsGolden: Boolean = false, | ||||
|                          var slabsHighBank: Int? = null, | ||||
|                          var slabsGolden: Boolean = false, | ||||
|                          var dontSplitWordArrays: Boolean = false, | ||||
|                          var breakpointCpuInstruction: String? = null, | ||||
|                          var ignoreFootguns: Boolean = false, | ||||
|                          var outputDir: Path = Path(""), | ||||
|                          var quiet: Boolean = false, | ||||
|                          var symbolDefs: Map<String, String> = emptyMap() | ||||
| ) { | ||||
|     init { | ||||
|         compTarget.initializeMemoryAreas(this) | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         val AllZeropageAllowed: List<UIntRange> = listOf(0u..255u) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										97
									
								
								codeCore/src/prog8/code/core/Conversions.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										97
									
								
								codeCore/src/prog8/code/core/Conversions.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,97 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| 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 { | ||||
|     //  0..15 -> "0".."15" | ||||
|     //  16..255 -> "$10".."$ff" | ||||
|     //  256..65536 -> "$0100".."$ffff" | ||||
|     //  larger -> "$12345678" | ||||
|     // negative values are prefixed with '-'. | ||||
|     val integer = this.toInt() | ||||
|     if(integer<0) | ||||
|         return '-' + abs(integer).toHex() | ||||
|     return when (integer) { | ||||
|         in 0 until 16 -> integer.toString() | ||||
|         in 0 until 0x100 -> "$"+integer.toString(16).padStart(2,'0') | ||||
|         in 0 until 0x10000 -> "$"+integer.toString(16).padStart(4,'0') | ||||
|         else -> "$"+integer.toString(16).padStart(8,'0') | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun UInt.toHex(): String { | ||||
|     //  0..15 -> "0".."15" | ||||
|     //  16..255 -> "$10".."$ff" | ||||
|     //  256..65536 -> "$0100".."$ffff" | ||||
|     //  larger -> "$12345678" | ||||
|     return when (this) { | ||||
|         in 0u until 16u -> this.toString() | ||||
|         in 0u until 0x100u -> "$"+this.toString(16).padStart(2,'0') | ||||
|         in 0u until 0x10000u -> "$"+this.toString(16).padStart(4,'0') | ||||
|         else -> "$"+this.toString(16).padStart(8,'0') | ||||
|     } | ||||
| } | ||||
|  | ||||
| fun Char.escape(): Char = this.toString().escape()[0] | ||||
|  | ||||
| fun String.escape(): String { | ||||
|     val es = this.map { | ||||
|         when(it) { | ||||
|             '\t' -> "\\t" | ||||
|             '\n' -> "\\n" | ||||
|             '\r' -> "\\r" | ||||
|             '"' -> "\\\"" | ||||
|             in '\u8000'..'\u80ff' -> "\\x" + (it.code - 0x8000).toString(16).padStart(2, '0')       // 'ugly' passthrough hack | ||||
|             in '\u0000'..'\u00ff' -> it.toString() | ||||
|             else -> "\\u" + it.code.toString(16).padStart(4, '0') | ||||
|         } | ||||
|     } | ||||
|     return es.joinToString("") | ||||
| } | ||||
|  | ||||
| fun String.unescape(): String { | ||||
|     val result = mutableListOf<Char>() | ||||
|     val iter = this.iterator() | ||||
|     while(iter.hasNext()) { | ||||
|         val c = iter.nextChar() | ||||
|         if(c=='\\') { | ||||
|             val ec = iter.nextChar() | ||||
|             result.add(when(ec) { | ||||
|                 '\\' -> '\\' | ||||
|                 'n' -> '\n' | ||||
|                 'r' -> '\r' | ||||
|                 't' -> '\t' | ||||
|                 '"' -> '"' | ||||
|                 '\'' -> '\'' | ||||
|                 'u' -> { | ||||
|                     try { | ||||
|                         "${iter.nextChar()}${iter.nextChar()}${iter.nextChar()}${iter.nextChar()}".toInt(16).toChar() | ||||
|                     } catch (sb: StringIndexOutOfBoundsException) { | ||||
|                         throw IllegalArgumentException("invalid \\u escape sequence") | ||||
|                     } catch (nf: NumberFormatException) { | ||||
|                         throw IllegalArgumentException("invalid \\u escape sequence") | ||||
|                     } | ||||
|                 } | ||||
|                 'x' -> { | ||||
|                     try { | ||||
|                         val hex = ("" + iter.nextChar() + iter.nextChar()).toInt(16) | ||||
|                         (0x8000 + hex).toChar()         // 'ugly' pass-through hack | ||||
|                     } catch (sb: StringIndexOutOfBoundsException) { | ||||
|                         throw IllegalArgumentException("invalid \\x escape sequence") | ||||
|                     } catch (nf: NumberFormatException) { | ||||
|                         throw IllegalArgumentException("invalid \\x escape sequence") | ||||
|                     } | ||||
|                 } | ||||
|                 else -> throw IllegalArgumentException("invalid escape char in string: \\$ec") | ||||
|             }) | ||||
|         } else { | ||||
|             result.add(c) | ||||
|         } | ||||
|     } | ||||
|     return result.joinToString("") | ||||
| } | ||||
							
								
								
									
										503
									
								
								codeCore/src/prog8/code/core/Enumerations.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										503
									
								
								codeCore/src/prog8/code/core/Enumerations.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,503 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| import java.util.* | ||||
|  | ||||
| enum class BaseDataType { | ||||
|     UBYTE,              // pass by value            8 bits unsigned | ||||
|     BYTE,               // pass by value            8 bits signed | ||||
|     UWORD,              // pass by value            16 bits unsigned | ||||
|     WORD,               // pass by value            16 bits signed | ||||
|     LONG,               // pass by value            32 bits signed | ||||
|     FLOAT,              // pass by value            machine dependent | ||||
|     BOOL,               // pass by value            bit 0 of an 8-bit byte | ||||
|     STR,                // pass by reference | ||||
|     ARRAY,              // pass by reference, subtype is the element type | ||||
|     ARRAY_SPLITW,       // pass by reference, split word layout, subtype is the element type (restricted to word types) | ||||
|     POINTER,            // typed pointer, subtype is whatever type is pointed to | ||||
|     STRUCT_INSTANCE,    // the actual instance of a struct (not directly supported in the language yet, but we need its type) | ||||
|     ARRAY_POINTER,      // array of pointers (uwords), subtype is whatever type each element points to | ||||
|     UNDEFINED; | ||||
|  | ||||
|  | ||||
|     fun largerSizeThan(other: BaseDataType) = | ||||
|         when { | ||||
|             this == other -> false | ||||
|             this.isByteOrBool -> false | ||||
|             this.isWord -> other.isByteOrBool | ||||
|             this == LONG -> other.isByteOrBool || other.isWord | ||||
|             this == STR && other == UWORD || this == UWORD && other == STR -> false | ||||
|             this.isArray && other.isArray -> false | ||||
|             this.isArray -> other != FLOAT | ||||
|             this == STR -> other != FLOAT | ||||
|             this.isPointer -> other.isByteOrBool | ||||
|             else -> true | ||||
|         } | ||||
|  | ||||
|     fun equalsSize(other: BaseDataType) = | ||||
|         when { | ||||
|             this == other -> true | ||||
|             this.isArray && other.isArray -> true | ||||
|             this.isByteOrBool -> other.isByteOrBool | ||||
|             this.isWord -> other.isWord || other.isPointer | ||||
|             this.isPointer -> other.isWord | ||||
|             this == STR && other== UWORD || this== UWORD && other== STR -> true | ||||
|             this == STR && other.isArray -> true | ||||
|             this.isArray && other == STR -> true | ||||
|             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 || this == BaseDataType.ARRAY_POINTER | ||||
| val BaseDataType.isPointer get() = this == BaseDataType.POINTER | ||||
| val BaseDataType.isStructInstance get() = this == BaseDataType.STRUCT_INSTANCE | ||||
| val BaseDataType.isPointerArray get() = this == BaseDataType.ARRAY_POINTER | ||||
| val BaseDataType.isSplitWordArray get() = this == BaseDataType.ARRAY_SPLITW || this == BaseDataType.ARRAY_POINTER       // pointer arrays are also always stored as split uwords | ||||
| val BaseDataType.isIterable get() =  this in arrayOf(BaseDataType.STR, BaseDataType.ARRAY, BaseDataType.ARRAY_SPLITW, BaseDataType.ARRAY_POINTER) | ||||
| val BaseDataType.isPassByRef get() = this.isIterable && !this.isPointer | ||||
| val BaseDataType.isPassByValue get() = !this.isIterable || this.isPointer | ||||
|  | ||||
|  | ||||
| interface ISubType { | ||||
|     val scopedNameString: String | ||||
|     fun memsize(sizer: IMemSizer): Int | ||||
|     fun sameas(other: ISubType): Boolean | ||||
|     fun getFieldType(name: String): DataType? | ||||
| } | ||||
|  | ||||
| class DataType private constructor(val base: BaseDataType, val sub: BaseDataType?, var subType: ISubType?, var subTypeFromAntlr: List<String>?=null) { | ||||
|  | ||||
|     init { | ||||
|         when { | ||||
|             base.isPointerArray -> { | ||||
|                 require(sub!=null || subType!=null || subTypeFromAntlr!=null) | ||||
|             } | ||||
|             base.isArray -> { | ||||
|                 require(sub != null && subType==null && subTypeFromAntlr==null) | ||||
|                 if(base.isSplitWordArray) | ||||
|                     require(sub == BaseDataType.UWORD || sub == BaseDataType.WORD) | ||||
|             } | ||||
|             base==BaseDataType.STR -> require(sub==BaseDataType.UBYTE) { "string subtype should be ubyte" } | ||||
|             base!=BaseDataType.POINTER -> require(sub == null) { "only string, array and pointer base types can have a subtype"} | ||||
|             else -> { | ||||
|                 require(sub == null || (subType == null && subTypeFromAntlr == null)) { | ||||
|                     "sub and subtype can't both be set" | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun equals(other: Any?): Boolean { | ||||
|         if (this === other) return true | ||||
|         if (other !is DataType) return false | ||||
|         return base == other.base && sub == other.sub && (subType==other.subType || subType!!.sameas(other.subType!!)) | ||||
|     } | ||||
|  | ||||
|     override fun hashCode(): Int = Objects.hash(base, sub, subType) | ||||
|  | ||||
|     fun setActualSubType(actualSubType: ISubType) { | ||||
|         subType = actualSubType | ||||
|         subTypeFromAntlr = null | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|  | ||||
|         val UBYTE = DataType(BaseDataType.UBYTE, null, null) | ||||
|         val BYTE = DataType(BaseDataType.BYTE, null, null) | ||||
|         val UWORD = DataType(BaseDataType.UWORD, null, null) | ||||
|         val WORD = DataType(BaseDataType.WORD, null, null) | ||||
|         val LONG = DataType(BaseDataType.LONG, null, null) | ||||
|         val FLOAT = DataType(BaseDataType.FLOAT, null, null) | ||||
|         val BOOL = DataType(BaseDataType.BOOL, null, null) | ||||
|         val STR = DataType(BaseDataType.STR, BaseDataType.UBYTE, null) | ||||
|         val UNDEFINED = DataType(BaseDataType.UNDEFINED, null, null) | ||||
|  | ||||
|         private val simpletypes = mapOf( | ||||
|             BaseDataType.UBYTE to DataType(BaseDataType.UBYTE, null, null), | ||||
|             BaseDataType.BYTE to DataType(BaseDataType.BYTE, null, null), | ||||
|             BaseDataType.UWORD to DataType(BaseDataType.UWORD, null, null), | ||||
|             BaseDataType.WORD to DataType(BaseDataType.WORD, null, null), | ||||
|             BaseDataType.LONG to DataType(BaseDataType.LONG, null, null), | ||||
|             BaseDataType.FLOAT to DataType(BaseDataType.FLOAT, null, null), | ||||
|             BaseDataType.BOOL to DataType(BaseDataType.BOOL, null, null), | ||||
|             BaseDataType.STR to DataType(BaseDataType.STR, BaseDataType.UBYTE, null), | ||||
|             BaseDataType.UNDEFINED to DataType(BaseDataType.UNDEFINED, null, null) | ||||
|         ) | ||||
|  | ||||
|         fun forDt(dt: BaseDataType): DataType { | ||||
|             if(dt.isStructInstance) | ||||
|                 TODO("cannot use struct instance as a data type (yet) - use a pointer instead") | ||||
|             return simpletypes.getValue(dt) | ||||
|         } | ||||
|  | ||||
|         fun arrayFor(elementDt: BaseDataType, splitwordarray: Boolean=true): DataType { | ||||
|             require(!elementDt.isPointer) { "use other array constructor for arrays of pointers" } | ||||
|             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, null) | ||||
|             else { | ||||
|                 if(actualElementDt.isNumericOrBool && actualElementDt != BaseDataType.LONG) | ||||
|                     DataType(BaseDataType.ARRAY, actualElementDt, null) | ||||
|                 else | ||||
|                     throw NoSuchElementException("invalid basic element dt $elementDt") | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fun arrayOfPointersTo(sub: BaseDataType): DataType = DataType(BaseDataType.ARRAY_POINTER, sub, null) | ||||
|         fun arrayOfPointersTo(structType: ISubType?): DataType = DataType(BaseDataType.ARRAY_POINTER, null, structType) | ||||
|         fun arrayOfPointersFromAntlrTo(sub: BaseDataType?, identifier: List<String>?): DataType = | ||||
|             DataType(BaseDataType.ARRAY_POINTER, sub, null, identifier) | ||||
|  | ||||
|         fun pointer(base: BaseDataType): DataType = DataType(BaseDataType.POINTER, base, null) | ||||
|         fun pointer(dt: DataType): DataType = if(dt.isBasic) | ||||
|                 DataType(BaseDataType.POINTER, dt.base, null) | ||||
|             else | ||||
|                 DataType(BaseDataType.POINTER, null, dt.subType, dt.subTypeFromAntlr) | ||||
|         fun pointer(structType: ISubType): DataType = DataType(BaseDataType.POINTER, null, structType) | ||||
|         fun pointerFromAntlr(identifier: List<String>): DataType = DataType(BaseDataType.POINTER, null, null, identifier) | ||||
|         fun structInstance(type: ISubType?): DataType = DataType(BaseDataType.STRUCT_INSTANCE, sub=null, type) | ||||
|         fun structInstanceFromAntlr(struct: List<String>): DataType = DataType(BaseDataType.STRUCT_INSTANCE, null, null, subTypeFromAntlr = struct) | ||||
|     } | ||||
|  | ||||
|  | ||||
|     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 = | ||||
|         when { | ||||
|             isPointerArray -> DataType(BaseDataType.POINTER, sub, subType) | ||||
|             base.isArray || base==BaseDataType.STR -> forDt(sub!!) | ||||
|             else -> throw IllegalArgumentException("not an array") | ||||
|         } | ||||
|  | ||||
|     fun typeForAddressOf(msb: Boolean): DataType { | ||||
|         if (isUndefined) | ||||
|             return if(msb) pointer(BaseDataType.UBYTE) else UWORD | ||||
|         else { | ||||
|             if (isBasic) | ||||
|                 return pointer(base) | ||||
|             if (isString) | ||||
|                 return pointer(BaseDataType.UBYTE) | ||||
|             if (isPointer) | ||||
|                 return UWORD | ||||
|             if (isArray) { | ||||
|                 if (msb || isSplitWordArray) | ||||
|                     return pointer(BaseDataType.UBYTE) | ||||
|                 val elementDt = elementType() | ||||
|                 require(elementDt.isBasic) | ||||
|                 return pointer(elementDt) | ||||
|             } | ||||
|             if (subType != null) | ||||
|                 return pointer(this) | ||||
|             return UWORD | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun dereference(): DataType { | ||||
|         require(isPointer || isUnsignedWord) | ||||
|         return when { | ||||
|             isUnsignedWord -> forDt(BaseDataType.UBYTE) | ||||
|             sub!=null -> forDt(sub) | ||||
|             subType!=null -> DataType(BaseDataType.STRUCT_INSTANCE, null, subType) | ||||
|             subTypeFromAntlr!=null -> DataType(BaseDataType.STRUCT_INSTANCE, null, null, subTypeFromAntlr) | ||||
|             else -> throw IllegalArgumentException("cannot dereference this pointer type") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     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") | ||||
|             } | ||||
|         } | ||||
|         BaseDataType.POINTER -> { | ||||
|             if(sub!=null) "^^${sub.name.lowercase()}" else if(subType!=null) "^^${subType!!.scopedNameString}" else "^^${subTypeFromAntlr}" | ||||
|         } | ||||
|         BaseDataType.ARRAY_POINTER -> { | ||||
|             if(sub!=null) "^^${sub.name.lowercase()}[] (split)" else if (subType!=null) "^^${subType!!.scopedNameString}[] (split)" else "^^${subTypeFromAntlr}[] (split)" | ||||
|         } | ||||
|         BaseDataType.STRUCT_INSTANCE -> { | ||||
|             sub?.name?.lowercase() ?: if (subType!=null) subType!!.scopedNameString else "$subTypeFromAntlr" | ||||
|         } | ||||
|         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.POINTER -> { | ||||
|             when { | ||||
|                 sub!=null -> "^^${sub.name.lowercase()}" | ||||
|                 subType!=null -> "^^${subType!!.scopedNameString}" | ||||
|                 subTypeFromAntlr!=null -> "^^${subTypeFromAntlr!!.joinToString(".")}" | ||||
|                 else -> "?????" | ||||
|             } | ||||
|         } | ||||
|         BaseDataType.STRUCT_INSTANCE -> { | ||||
|             when { | ||||
|                 sub!=null -> sub.name.lowercase() | ||||
|                 subType!=null -> subType!!.scopedNameString | ||||
|                 subTypeFromAntlr!=null -> subTypeFromAntlr!!.joinToString(".") | ||||
|                 else -> "?????" | ||||
|             } | ||||
|         } | ||||
|         BaseDataType.ARRAY_POINTER -> { | ||||
|             when { | ||||
|                 sub!=null -> "^^${sub.name.lowercase()}[" | ||||
|                 subType!=null -> "^^${subType!!.scopedNameString}[" | ||||
|                 subTypeFromAntlr!=null -> "^^${subTypeFromAntlr!!.joinToString(".")}[" | ||||
|                 else -> "????? [" | ||||
|             } | ||||
|         } | ||||
|         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.POINTER, BaseDataType.ARRAY_POINTER) | ||||
|             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) || (targetType.isPointer && targetType.sub==BaseDataType.UBYTE) | ||||
|             BaseDataType.ARRAY, BaseDataType.ARRAY_SPLITW -> targetType.base in arrayOf(BaseDataType.ARRAY, BaseDataType.ARRAY_SPLITW) && targetType.sub == sub | ||||
|             BaseDataType.POINTER -> { | ||||
|                 when { | ||||
|                     targetType.base == BaseDataType.UWORD || targetType.base == BaseDataType.LONG -> true | ||||
|                     targetType.isPointer -> this.isUnsignedWord || this == targetType | ||||
|                     else -> false | ||||
|                 } | ||||
|             } | ||||
|             BaseDataType.STRUCT_INSTANCE -> false        // we cannot deal with actual struct instances yet in any shape or form (only getting fields from it) | ||||
|             BaseDataType.ARRAY_POINTER -> false | ||||
|             BaseDataType.UNDEFINED -> false | ||||
|         } | ||||
|  | ||||
|     fun largerSizeThan(other: DataType): Boolean = base.largerSizeThan(other.base) | ||||
|     fun equalsSize(other: DataType): Boolean = base.equalsSize(other.base) | ||||
|  | ||||
|     // note: for pointer types, size() doesn't return the size of the pointer itself but the size of the thing it points to | ||||
|     fun size(memsizer: IMemSizer): Int = if(sub!=null) { | ||||
|             memsizer.memorySize(sub) | ||||
|         } else if(subType!=null) { | ||||
|             subType!!.memsize(memsizer) | ||||
|         } else { | ||||
|             memsizer.memorySize(base) | ||||
|         } | ||||
|  | ||||
|     val isBasic = sub==null && subType==null && subTypeFromAntlr==null | ||||
|     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 isPointer = base.isPointer | ||||
|     val isPointerToByte = base.isPointer && sub?.isByteOrBool==true | ||||
|     val isPointerToWord = base.isPointer && sub?.isWord==true | ||||
|     val isStructInstance = base.isStructInstance | ||||
|     val isPointerArray = base.isPointerArray | ||||
|     val isBoolArray = base.isArray && !base.isPointerArray && sub == BaseDataType.BOOL | ||||
|     val isByteArray = base.isArray && !base.isPointerArray && (sub == BaseDataType.UBYTE || sub == BaseDataType.BYTE) | ||||
|     val isUnsignedByteArray = base.isArray && !base.isPointerArray && sub == BaseDataType.UBYTE | ||||
|     val isSignedByteArray = base.isArray && !base.isPointerArray && sub == BaseDataType.BYTE | ||||
|     val isWordArray = base.isArray && !base.isPointerArray && (sub == BaseDataType.UWORD || sub == BaseDataType.WORD) | ||||
|     val isUnsignedWordArray = base.isArray && !base.isPointerArray && sub == BaseDataType.UWORD | ||||
|     val isSignedWordArray = base.isArray && !base.isPointerArray && sub == BaseDataType.WORD | ||||
|     val isFloatArray = base.isArray && !base.isPointerArray && 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 && !base.isPointerArray && sub == BaseDataType.UWORD | ||||
|     val isSplitSignedWordArray = base.isSplitWordArray && !base.isPointerArray && sub == BaseDataType.WORD | ||||
|     val isIterable =  base.isIterable | ||||
|     val isPassByRef = base.isPassByRef | ||||
|     val isPassByValue = base.isPassByValue | ||||
| } | ||||
|  | ||||
|  | ||||
| enum class CpuRegister { | ||||
|     A, | ||||
|     X, | ||||
|     Y | ||||
| } | ||||
|  | ||||
| enum class RegisterOrPair { | ||||
|     A, | ||||
|     X, | ||||
|     Y, | ||||
|     AX, | ||||
|     AY, | ||||
|     XY, | ||||
|     FAC1, | ||||
|     FAC2, | ||||
|     // cx16 virtual registers: | ||||
|     R0, R1, R2, R3, R4, R5, R6, R7, | ||||
|     R8, R9, R10, R11, R12, R13, R14, R15; | ||||
|  | ||||
|     companion object { | ||||
|         val names by lazy { entries.map { it.toString()} } | ||||
|         fun fromCpuRegister(cpu: CpuRegister): RegisterOrPair { | ||||
|             return when(cpu) { | ||||
|                 CpuRegister.A -> A | ||||
|                 CpuRegister.X -> X | ||||
|                 CpuRegister.Y -> Y | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun asCpuRegister(): CpuRegister = when(this) { | ||||
|         A -> CpuRegister.A | ||||
|         X -> CpuRegister.X | ||||
|         Y -> CpuRegister.Y | ||||
|         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 IllegalArgumentException("invalid register param type") | ||||
|         } | ||||
|         return listOf("cx16", name.lowercase()+suffix) | ||||
|     } | ||||
|  | ||||
|     fun isWord() = this==AX || this == AY || this==XY || this in Cx16VirtualRegisters | ||||
|  | ||||
| }       // only used in parameter and return value specs in asm subroutines | ||||
|  | ||||
| enum class Statusflag { | ||||
|     Pc, | ||||
|     Pz,     // don't use | ||||
|     Pv, | ||||
|     Pn;     // don't use | ||||
|  | ||||
|     companion object { | ||||
|         val names by lazy { entries.map { it.toString()} } | ||||
|     } | ||||
| } | ||||
|  | ||||
| enum class BranchCondition { | ||||
|     CS, | ||||
|     CC, | ||||
|     EQ,     // EQ == Z | ||||
|     Z, | ||||
|     NE,     // NE == NZ | ||||
|     NZ, | ||||
|     MI,     // MI == NEG | ||||
|     NEG, | ||||
|     PL,     // PL == POS | ||||
|     POS, | ||||
|     VS, | ||||
|     VC | ||||
| } | ||||
|  | ||||
| val Cx16VirtualRegisters = arrayOf( | ||||
|     RegisterOrPair.R0, RegisterOrPair.R1, RegisterOrPair.R2, RegisterOrPair.R3, | ||||
|     RegisterOrPair.R4, RegisterOrPair.R5, RegisterOrPair.R6, RegisterOrPair.R7, | ||||
|     RegisterOrPair.R8, RegisterOrPair.R9, RegisterOrPair.R10, RegisterOrPair.R11, | ||||
|     RegisterOrPair.R12, RegisterOrPair.R13, RegisterOrPair.R14, RegisterOrPair.R15 | ||||
| ) | ||||
|  | ||||
| val CpuRegisters = arrayOf( | ||||
|     RegisterOrPair.A, RegisterOrPair.X, RegisterOrPair.Y, | ||||
|     RegisterOrPair.AX, RegisterOrPair.AY, RegisterOrPair.XY | ||||
| ) | ||||
|  | ||||
|  | ||||
| enum class OutputType { | ||||
|     RAW, | ||||
|     PRG, | ||||
|     XEX, | ||||
|     LIBRARY | ||||
| } | ||||
|  | ||||
| enum class CbmPrgLauncherType { | ||||
|     BASIC, | ||||
|     NONE | ||||
| } | ||||
|  | ||||
| enum class ZeropageType { | ||||
|     BASICSAFE, | ||||
|     FLOATSAFE, | ||||
|     KERNALSAFE, | ||||
|     FULL, | ||||
|     DONTUSE | ||||
| } | ||||
|  | ||||
| enum class ZeropageWish { | ||||
|     REQUIRE_ZEROPAGE, | ||||
|     PREFER_ZEROPAGE, | ||||
|     DONTCARE, | ||||
|     NOT_IN_ZEROPAGE | ||||
| } | ||||
|  | ||||
| enum class SplitWish { | ||||
|     DONTCARE, | ||||
|     SPLIT, | ||||
|     NOSPLIT | ||||
| } | ||||
							
								
								
									
										7
									
								
								codeCore/src/prog8/code/core/Exceptions.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								codeCore/src/prog8/code/core/Exceptions.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| class InternalCompilerException(message: String?) : Exception(message) | ||||
|  | ||||
| class AssemblyError(msg: String) : RuntimeException(msg) | ||||
|  | ||||
| class ErrorsReportedException(message: String?) : Exception(message) | ||||
							
								
								
									
										44
									
								
								codeCore/src/prog8/code/core/ICompilationTarget.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										44
									
								
								codeCore/src/prog8/code/core/ICompilationTarget.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,44 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| import java.nio.file.Path | ||||
|  | ||||
| enum class CpuType { | ||||
|     CPU6502, | ||||
|     CPU65C02, | ||||
|     VIRTUAL | ||||
| } | ||||
|  | ||||
| interface ICompilationTarget: IStringEncoding, IMemSizer { | ||||
|     val name: String | ||||
|  | ||||
|     val FLOAT_MAX_NEGATIVE: Double | ||||
|     val FLOAT_MAX_POSITIVE: Double | ||||
|     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_MEMTOP_ADDRESS: UInt | ||||
|     val BSSHIGHRAM_START: UInt | ||||
|     val BSSHIGHRAM_END: UInt | ||||
|     val BSSGOLDENRAM_START: UInt | ||||
|     val BSSGOLDENRAM_END: UInt | ||||
|  | ||||
|     val cpu: CpuType | ||||
|     var zeropage: Zeropage | ||||
|     var golden: GoldenRam | ||||
|     val libraryPath: Path? | ||||
|     val customLauncher: List<String> | ||||
|     val additionalAssemblerOptions: List<String> | ||||
|     val defaultOutputType: OutputType | ||||
|  | ||||
|     fun initializeMemoryAreas(compilerOptions: CompilationOptions) | ||||
|     fun getFloatAsmBytes(num: Number): String | ||||
|  | ||||
|     fun convertFloatToBytes(num: Double): List<UByte> | ||||
|     fun convertBytesToFloat(bytes: List<UByte>): Double | ||||
|  | ||||
|     fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path, quiet: Boolean) | ||||
|     fun isIOAddress(address: UInt): Boolean | ||||
|  | ||||
|     override fun encodeString(str: String, encoding: Encoding): List<UByte> | ||||
|     override fun decodeString(bytes: Iterable<UByte>, encoding: Encoding): String | ||||
| } | ||||
							
								
								
									
										18
									
								
								codeCore/src/prog8/code/core/IErrorReporter.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								codeCore/src/prog8/code/core/IErrorReporter.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| interface IErrorReporter { | ||||
|     fun err(msg: String, position: Position) | ||||
|     fun warn(msg: String, position: Position) | ||||
|     fun info(msg: String, position: Position) | ||||
|     fun undefined(symbol: List<String>, position: Position) | ||||
|     fun noErrors(): Boolean | ||||
|     fun report() | ||||
|     fun finalizeNumErrors(numErrors: Int, numWarnings: Int, numInfos: Int) { | ||||
|         if(numErrors>0) | ||||
|             throw ErrorsReportedException("There are $numErrors errors, $numWarnings warnings, and $numInfos infos.") | ||||
|     } | ||||
|  | ||||
|     fun noErrorForLine(position: Position): Boolean | ||||
|  | ||||
|     fun printSingleError(errormessage: String) | ||||
| } | ||||
							
								
								
									
										15
									
								
								codeCore/src/prog8/code/core/IMemSizer.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								codeCore/src/prog8/code/core/IMemSizer.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| interface IMemSizer { | ||||
|     fun memorySize(dt: DataType, numElements: Int?): Int | ||||
|  | ||||
|     fun memorySize(dt: BaseDataType): Int { | ||||
|         if(dt.isPassByRef) | ||||
|             return memorySize(DataType.UWORD, null)      // a pointer size | ||||
|         try { | ||||
|             return memorySize(DataType.forDt(dt), null) | ||||
|         } catch (x: NoSuchElementException) { | ||||
|             throw IllegalArgumentException(x.message) | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										21
									
								
								codeCore/src/prog8/code/core/IStringEncoding.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								codeCore/src/prog8/code/core/IStringEncoding.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| enum class Encoding(val prefix: String) { | ||||
|     DEFAULT("default"),         // depends on compilation target | ||||
|     PETSCII("petscii"),         // c64/c128/cx16 | ||||
|     SCREENCODES("sc"),          // c64/c128/cx16 | ||||
|     ATASCII("atascii"),         // atari | ||||
|     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) | ||||
|     C64OS("c64os")              // c64 (C64 OS) | ||||
| } | ||||
|  | ||||
| interface IStringEncoding { | ||||
|     val defaultEncoding: Encoding | ||||
|  | ||||
|     fun encodeString(str: String, encoding: Encoding): List<UByte> | ||||
|     fun decodeString(bytes: Iterable<UByte>, encoding: Encoding): String | ||||
| } | ||||
							
								
								
									
										173
									
								
								codeCore/src/prog8/code/core/MemoryRegions.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										173
									
								
								codeCore/src/prog8/code/core/MemoryRegions.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,173 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| import com.github.michaelbull.result.Err | ||||
| import com.github.michaelbull.result.Ok | ||||
| import com.github.michaelbull.result.Result | ||||
|  | ||||
|  | ||||
| class MemAllocationError(message: String) : Exception(message) | ||||
|  | ||||
|  | ||||
| abstract class MemoryAllocator(protected val options: CompilationOptions) { | ||||
|     data class VarAllocation(val address: UInt, val dt: DataType, val size: Int) | ||||
|  | ||||
|     abstract fun allocate(name: String, | ||||
|                           datatype: DataType, | ||||
|                           numElements: Int?, | ||||
|                           position: Position?, | ||||
|                           errors: IErrorReporter): Result<VarAllocation, MemAllocationError> | ||||
| } | ||||
|  | ||||
|  | ||||
| abstract class Zeropage(options: CompilationOptions): MemoryAllocator(options) { | ||||
|  | ||||
|     abstract val SCRATCH_B1 : UInt      // temp storage for a single byte | ||||
|     abstract val SCRATCH_REG : UInt     // temp storage for a register byte, must be B1+1 | ||||
|     abstract val SCRATCH_W1 : UInt      // temp storage 1 for a word | ||||
|     abstract val SCRATCH_W2 : UInt      // temp storage 2 for a word | ||||
|     abstract val SCRATCH_PTR : UInt     // temp storage for a pointer | ||||
|  | ||||
|  | ||||
|     // the variables allocated into Zeropage. | ||||
|     // name (scoped) ==> pair of address to (Datatype + bytesize) | ||||
|     val allocatedVariables = mutableMapOf<String, VarAllocation>() | ||||
|  | ||||
|     val free = mutableListOf<UInt>()     // subclasses must set this to the appropriate free locations. | ||||
|  | ||||
|     fun removeReservedFromFreePool() { | ||||
|         synchronized(this) { | ||||
|             for (reserved in options.zpReserved) | ||||
|                 reserve(reserved) | ||||
|  | ||||
|             free.removeAll(arrayOf(SCRATCH_B1, SCRATCH_REG, SCRATCH_W1, SCRATCH_W1 + 1u, SCRATCH_W2, SCRATCH_W2 + 1u, SCRATCH_PTR, SCRATCH_PTR+1u)) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun retainAllowed() { | ||||
|         synchronized(this) { | ||||
|             for(allowed in options.zpAllowed) | ||||
|                 free.retainAll { it in allowed } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun availableBytes() = if(options.zeropage== ZeropageType.DONTUSE) 0 else free.size | ||||
|     fun hasByteAvailable() = if(options.zeropage== ZeropageType.DONTUSE) false else free.isNotEmpty() | ||||
|     fun hasWordAvailable(): Boolean { | ||||
|         if(options.zeropage== ZeropageType.DONTUSE) | ||||
|             return false | ||||
|  | ||||
|         return free.windowed(2).any { it[0] == it[1] - 1u } | ||||
|     } | ||||
|  | ||||
|     override fun allocate(name: String, | ||||
|                           datatype: DataType, | ||||
|                           numElements: Int?, | ||||
|                           position: Position?, | ||||
|                           errors: IErrorReporter): Result<VarAllocation, MemAllocationError> { | ||||
|  | ||||
|         require(name.isEmpty() || name !in allocatedVariables) {"name can't be allocated twice"} | ||||
|  | ||||
|         if(options.zeropage== ZeropageType.DONTUSE) | ||||
|             return Err(MemAllocationError("zero page usage has been disabled")) | ||||
|  | ||||
|         val size: Int = | ||||
|                 when { | ||||
|                     datatype.isIntegerOrBool -> options.compTarget.memorySize(datatype, null) | ||||
|                     datatype.isPointer -> options.compTarget.memorySize(datatype, null) | ||||
|                     datatype.isString || datatype.isArray -> { | ||||
|                         val memsize = options.compTarget.memorySize(datatype, numElements!!) | ||||
|                         if(position!=null) | ||||
|                             errors.warn("allocating a large value in zeropage; str/array $memsize bytes", position) | ||||
|                         else | ||||
|                             errors.warn("$name: allocating a large value in zeropage; str/array $memsize bytes", Position.DUMMY) | ||||
|                         memsize | ||||
|                     } | ||||
|                     datatype.isFloat -> { | ||||
|                         if (options.floats) { | ||||
|                             val memsize = options.compTarget.memorySize(DataType.FLOAT, null) | ||||
|                             if(position!=null) | ||||
|                                 errors.warn("allocating a large value in zeropage; float $memsize bytes", position) | ||||
|                             else | ||||
|                                 errors.warn("$name: allocating a large value in zeropage; float $memsize bytes", Position.DUMMY) | ||||
|                             memsize | ||||
|                         } else return Err(MemAllocationError("floating point option not enabled")) | ||||
|                     } | ||||
|                     else -> throw MemAllocationError("weird dt") | ||||
|                 } | ||||
|  | ||||
|         synchronized(this) { | ||||
|             if(free.isNotEmpty()) { | ||||
|                 if(size==1) { | ||||
|                     for(candidate in free.minOrNull()!! .. free.maxOrNull()!!+1u) { | ||||
|                         if(oneSeparateByteFree(candidate)) | ||||
|                             return Ok(VarAllocation(makeAllocation(candidate, 1, datatype, name), datatype,1)) | ||||
|                     } | ||||
|                     return Ok(VarAllocation(makeAllocation(free[0], 1, datatype, name), datatype,1)) | ||||
|                 } | ||||
|                 for(candidate in free.minOrNull()!! .. free.maxOrNull()!!+1u) { | ||||
|                     if (sequentialFree(candidate, size)) | ||||
|                         return Ok(VarAllocation(makeAllocation(candidate, size, datatype, name), datatype, size)) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return Err(MemAllocationError("no more free space in ZP to allocate $size sequential bytes")) | ||||
|     } | ||||
|  | ||||
|     private fun reserve(range: UIntRange) = free.removeAll(range) | ||||
|  | ||||
|     private fun makeAllocation(address: UInt, size: Int, datatype: DataType, name: String): UInt { | ||||
|         require(size>=0) | ||||
|         free.removeAll(address until address+size.toUInt()) | ||||
|         if(name.isNotEmpty()) { | ||||
|             allocatedVariables[name] = when { | ||||
|                 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.isString -> VarAllocation(address, datatype, size) | ||||
|                 datatype.isArray -> VarAllocation(address, datatype, size) | ||||
|                 datatype.isPointer -> VarAllocation(address, datatype, size) | ||||
|                 else -> throw AssemblyError("invalid dt") | ||||
|             } | ||||
|         } | ||||
|         return address | ||||
|     } | ||||
|  | ||||
|     private fun oneSeparateByteFree(address: UInt) = address in free && address-1u !in free && address+1u !in free | ||||
|     private fun sequentialFree(address: UInt, size: Int): Boolean { | ||||
|         require(size>0) | ||||
|         return free.containsAll((address until address+size.toUInt()).toList()) | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| // TODO: this class is not yet used | ||||
| class GoldenRam(options: CompilationOptions, val region: UIntRange): MemoryAllocator(options) { | ||||
|     private var nextLocation: UInt = region.first | ||||
|  | ||||
|     override fun allocate( | ||||
|         name: String, | ||||
|         datatype: DataType, | ||||
|         numElements: Int?, | ||||
|         position: Position?, | ||||
|         errors: IErrorReporter): Result<VarAllocation, MemAllocationError> { | ||||
|  | ||||
|         val size: Int = | ||||
|             when { | ||||
|                 datatype.isIntegerOrBool -> options.compTarget.memorySize(datatype, null) | ||||
|                 datatype.isString -> numElements!! | ||||
|                 datatype.isArray -> options.compTarget.memorySize(datatype, numElements!!) | ||||
|                 datatype.isFloat -> { | ||||
|                     if (options.floats) { | ||||
|                         options.compTarget.memorySize(DataType.FLOAT, null) | ||||
|                     } else return Err(MemAllocationError("floating point option not enabled")) | ||||
|                 } | ||||
|                 else -> throw MemAllocationError("weird dt") | ||||
|             } | ||||
|  | ||||
|         return if(nextLocation<=region.last && (region.last + 1u - nextLocation) >= size.toUInt()) { | ||||
|             val result = Ok(VarAllocation(nextLocation, datatype, size)) | ||||
|             nextLocation += size.toUInt() | ||||
|             result | ||||
|         } else | ||||
|             Err(MemAllocationError("no more free space in Golden RAM to allocate $size sequential bytes")) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										18
									
								
								codeCore/src/prog8/code/core/Operators.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								codeCore/src/prog8/code/core/Operators.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| val AssociativeOperators = arrayOf("+", "*", "&", "|", "^", "==", "!=", "xor")       // note: and,or are not associative because of Shortcircuit/McCarthy evaluation | ||||
| val ComparisonOperators = arrayOf("==", "!=", "<", ">", "<=", ">=") | ||||
| val LogicalOperators = arrayOf("and", "or", "xor", "not", "in") | ||||
| val BitwiseOperators = arrayOf("&", "|", "^", "~") | ||||
| val PrefixOperators = arrayOf("+", "-", "~", "not") | ||||
|  | ||||
| fun invertedComparisonOperator(operator: String) = | ||||
|     when (operator) { | ||||
|         "==" -> "!=" | ||||
|         "!=" -> "==" | ||||
|         "<" -> ">=" | ||||
|         ">" -> "<=" | ||||
|         "<=" -> ">" | ||||
|         ">=" -> "<" | ||||
|         else -> null | ||||
|     } | ||||
							
								
								
									
										27
									
								
								codeCore/src/prog8/code/core/Position.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								codeCore/src/prog8/code/core/Position.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| import prog8.code.sanitize | ||||
| import prog8.code.source.SourceCode | ||||
| import java.nio.file.InvalidPathException | ||||
| import kotlin.io.path.Path | ||||
|  | ||||
| data class Position(val file: String, val line: Int, val startCol: Int, val endCol: Int) { | ||||
|     override fun toString(): String = "[$file: line $line col ${startCol+1}-${endCol+1}]" | ||||
|     fun toClickableStr(): String { | ||||
|         if(this===DUMMY) | ||||
|             return "" | ||||
|         if(SourceCode.isLibraryResource(file)) | ||||
|             return "$file:$line:$startCol:" | ||||
|         return try { | ||||
|             val path = Path(file).sanitize().toString() | ||||
|             "file://$path:$line:$startCol:" | ||||
|         } catch(_: InvalidPathException) { | ||||
|             // this can occur on Windows when the source origin contains "invalid" characters such as ':' | ||||
|             "file://$file:$line:$startCol:" | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         val DUMMY = Position("~dummy~", 0, 0, 0) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										3
									
								
								codeCore/src/prog8/code/core/RegisterOrStatusflag.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								codeCore/src/prog8/code/core/RegisterOrStatusflag.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | ||||
| package prog8.code.core | ||||
|  | ||||
| data class RegisterOrStatusflag(val registerOrPair: RegisterOrPair?, val statusflag: Statusflag?) | ||||
							
								
								
									
										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 prog8.code.sanitize | ||||
| import java.nio.file.Path | ||||
| import java.util.* | ||||
| import kotlin.io.path.Path | ||||
|  | ||||
|  | ||||
| // 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, isLibrary: Boolean=false): SourceCode { | ||||
|         val normalized = path.sanitize() | ||||
|         val cached = cache[normalized.toString()] | ||||
|         if (cached != null) | ||||
|             return cached | ||||
|         val file = SourceCode.File(normalized, isLibrary) | ||||
|         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).sanitize() | ||||
|         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>>() | ||||
| } | ||||
							
								
								
									
										161
									
								
								codeCore/src/prog8/code/source/SourceCode.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										161
									
								
								codeCore/src/prog8/code/source/SourceCode.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,161 @@ | ||||
| package prog8.code.source | ||||
|  | ||||
| import prog8.code.sanitize | ||||
| import java.io.IOException | ||||
| import java.nio.file.Path | ||||
| import java.text.Normalizer | ||||
| import kotlin.io.path.Path | ||||
| import kotlin.io.path.absolute | ||||
| import kotlin.io.path.readText | ||||
|  | ||||
| /** | ||||
|  * Encapsulates - and ties together - actual source code (=text) and its [origin]. | ||||
|  */ | ||||
| sealed class SourceCode { | ||||
|  | ||||
|     /** | ||||
|      * Whether this [SourceCode] instance was created as a [Resource] | ||||
|      */ | ||||
|     abstract val isFromResources: Boolean | ||||
|  | ||||
|     /** | ||||
|      * Whether this [SourceCode] instance was created as a [File] | ||||
|      */ | ||||
|     abstract val isFromFilesystem: Boolean | ||||
|  | ||||
|     /** | ||||
|      * Whether this [SourceCode] instance was created from a library module file | ||||
|      */ | ||||
|     abstract val isFromLibrary: Boolean | ||||
|  | ||||
|     /** | ||||
|      * The logical name of the source code unit. Usually the module's name. | ||||
|      */ | ||||
|     abstract val name: String | ||||
|  | ||||
|     /** | ||||
|      * Where this [SourceCode] instance came from. | ||||
|      * This can be one of the following: | ||||
|      * * a normal string representation of a [java.nio.file.Path], if it originates from a file (see [File]) | ||||
|      * * `string:44c56085` if was created via [String] | ||||
|      * * `library:/x/y/z.ext` if it is a library file that was loaded from resources (see [Resource]) | ||||
|      */ | ||||
|     abstract val origin: String | ||||
|  | ||||
|     /** | ||||
|      * The source code as plain string. | ||||
|      */ | ||||
|     abstract val text: String | ||||
|  | ||||
|     /** | ||||
|      * Printable representation, deliberately does NOT return the actual text. | ||||
|      */ | ||||
|     final override fun toString() = "${this.javaClass.name}[${this.origin}]" | ||||
|  | ||||
|     companion object { | ||||
|  | ||||
|         /** | ||||
|          * filename prefix to designate library files that will be retreived from internal resources rather than disk | ||||
|          */ | ||||
|         private const val LIBRARYFILEPREFIX = "library:" | ||||
|         private const val STRINGSOURCEPREFIX = "string:" | ||||
|         val curdir: Path = Path(".").absolute() | ||||
|         fun relative(path: Path): Path = curdir.relativize(path.sanitize()) | ||||
|         fun isRegularFilesystemPath(pathString: String) = !isLibraryResource(pathString) && !isStringResource(pathString) | ||||
|         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 | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Turn a plain String into a [SourceCode] object. | ||||
|      * [origin] will be something like `string:44c56085`. | ||||
|      */ | ||||
|     class Text(origText: String): SourceCode() { | ||||
|         override val text = origText.replace("\\R".toRegex(), "\n")      // normalize line endings | ||||
|         override val isFromResources = false | ||||
|         override val isFromFilesystem = false | ||||
|         override val isFromLibrary = false | ||||
|         override val origin = "$STRINGSOURCEPREFIX${System.identityHashCode(text).toString(16)}" | ||||
|         override val name = "<unnamed-text>" | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Get [SourceCode] from the file represented by the specified Path. | ||||
|      * 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. | ||||
|      * @throws NoSuchFileException if the file does not exist | ||||
|      * @throws FileSystemException if the file cannot be read | ||||
|      */ | ||||
|     internal class File(path: Path, override val isFromLibrary: Boolean): SourceCode() { | ||||
|         override val text: String | ||||
|         override val origin: String | ||||
|         override val name: String | ||||
|         override val isFromResources = false | ||||
|         override val isFromFilesystem = true | ||||
|  | ||||
|         init { | ||||
|             val normalized = path.normalize() | ||||
|             origin = relative(normalized).toString() | ||||
|             try { | ||||
|                 val contents = Normalizer.normalize(normalized.readText(), Normalizer.Form.NFC) | ||||
|                 text = contents.replace("\\R".toRegex(), "\n")      // normalize line endings | ||||
|                 name = normalized.toFile().nameWithoutExtension | ||||
|             } catch (nfx: java.nio.file.NoSuchFileException) { | ||||
|                 throw NoSuchFileException(normalized.toFile()).also { it.initCause(nfx) } | ||||
|             } catch (iox: IOException) { | ||||
|                 throw FileSystemException(normalized.toFile()).also { it.initCause(iox) } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * [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. | ||||
|      */ | ||||
|     internal class Resource(pathString: String): SourceCode() { | ||||
|         private val normalized = "/" + Path(pathString).normalize().toMutableList().joinToString("/") | ||||
|  | ||||
|         override val isFromResources = true | ||||
|         override val isFromFilesystem = false | ||||
|         override val isFromLibrary = true | ||||
|         override val origin = "$LIBRARYFILEPREFIX$normalized" | ||||
|         override val text: String | ||||
|         override val name: String | ||||
|  | ||||
|         init { | ||||
|             val rscURL = object {}.javaClass.getResource(normalized) | ||||
|             if (rscURL == null) { | ||||
|                 val rscRoot = object {}.javaClass.getResource("/") | ||||
|                 throw NoSuchFileException( | ||||
|                     java.io.File(normalized), | ||||
|                     reason = "looked in resources rooted at $rscRoot" | ||||
|                 ) | ||||
|             } | ||||
|             val stream = object {}.javaClass.getResourceAsStream(normalized) | ||||
|             val contents = stream!!.reader().use { Normalizer.normalize(it.readText(), Normalizer.Form.NFC) } | ||||
|             text = contents.replace("\\R".toRegex(), "\n")      // normalize line endings | ||||
|             name = Path(pathString).toFile().nameWithoutExtension | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * SourceCode for internally generated nodes (usually Modules) | ||||
|      */ | ||||
|     class Generated(override val name: String) : SourceCode() { | ||||
|         override val isFromResources: Boolean = false | ||||
|         override val isFromFilesystem: Boolean = false | ||||
|         override val isFromLibrary: Boolean = false | ||||
|         override val origin: String = name | ||||
|         override val text: String = "<generated code node, no text representation>" | ||||
|     } | ||||
| } | ||||
							
								
								
									
										81
									
								
								codeCore/src/prog8/code/target/C128Target.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										81
									
								
								codeCore/src/prog8/code/target/C128Target.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,81 @@ | ||||
| package prog8.code.target | ||||
|  | ||||
| import prog8.code.core.* | ||||
| import prog8.code.target.encodings.Encoder | ||||
| import prog8.code.target.zp.C128Zeropage | ||||
| import java.nio.file.Path | ||||
|  | ||||
|  | ||||
| class C128Target: ICompilationTarget, | ||||
|     IStringEncoding by Encoder(true), | ||||
|     IMemSizer by NormalMemSizer(Mflpt5.FLOAT_MEM_SIZE) { | ||||
|  | ||||
|     override val name = NAME | ||||
|     override val defaultEncoding = Encoding.PETSCII | ||||
|     override val libraryPath = null | ||||
|     override val customLauncher = emptyList<String>() | ||||
|     override val additionalAssemblerOptions = emptyList<String>() | ||||
|     override val defaultOutputType = OutputType.PRG | ||||
|  | ||||
|     companion object { | ||||
|         const val NAME = "c128" | ||||
|     } | ||||
|  | ||||
|  | ||||
|     override val cpu = CpuType.CPU6502 | ||||
|  | ||||
|     override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE | ||||
|     override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE | ||||
|     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_MEMTOP_ADDRESS = 0xc000u | ||||
|  | ||||
|     override val BSSHIGHRAM_START = 0u    // TODO address? | ||||
|     override val BSSHIGHRAM_END = 0u      // TODO address? | ||||
|     override val BSSGOLDENRAM_START = 0u  // TODO address? | ||||
|     override val BSSGOLDENRAM_END = 0u    // TODO address? | ||||
|  | ||||
|     override lateinit var zeropage: Zeropage | ||||
|     override lateinit var golden: GoldenRam | ||||
|  | ||||
|     override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm() | ||||
|  | ||||
|     override fun convertFloatToBytes(num: Double): List<UByte> { | ||||
|         val m5 = Mflpt5.fromNumber(num) | ||||
|         return listOf(m5.b0, m5.b1, m5.b2, m5.b3, m5.b4) | ||||
|     } | ||||
|  | ||||
|     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, quiet: Boolean) { | ||||
|         if(selectedEmulator!=1) { | ||||
|             System.err.println("The c128 target only supports the main emulator (Vice).") | ||||
|             return | ||||
|         } | ||||
|  | ||||
|         if(!quiet) | ||||
|             println("\nStarting C-128 emulator x128...") | ||||
|  | ||||
|         val viceMonlist = C64Target.viceMonListName(programNameWithPath.toString()) | ||||
|         val cmdline = listOf("x128", "-silent", "-moncommands", viceMonlist, | ||||
|             "-autostartprgmode", "1", "-autostart-warp", "-autostart", "${programNameWithPath}.prg") | ||||
|         val processb = ProcessBuilder(cmdline) | ||||
|         if(!quiet) | ||||
|             processb.inheritIO() | ||||
|         val process: Process = processb.start() | ||||
|         process.waitFor() | ||||
|     } | ||||
|  | ||||
|     override fun isIOAddress(address: UInt): Boolean = address==0u || address==1u || address in 0xd000u..0xdfffu | ||||
|  | ||||
|     override fun initializeMemoryAreas(compilerOptions: CompilationOptions) { | ||||
|         zeropage = C128Zeropage(compilerOptions) | ||||
|         golden = GoldenRam(compilerOptions, UIntRange.EMPTY)    // TODO does the c128 have some of this somewhere? | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										111
									
								
								codeCore/src/prog8/code/target/C64Target.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										111
									
								
								codeCore/src/prog8/code/target/C64Target.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,111 @@ | ||||
| package prog8.code.target | ||||
|  | ||||
| import prog8.code.core.* | ||||
| import prog8.code.target.encodings.Encoder | ||||
| import prog8.code.target.zp.C64Zeropage | ||||
| import java.io.IOException | ||||
| import java.nio.file.Path | ||||
|  | ||||
|  | ||||
| class C64Target: ICompilationTarget, | ||||
|     IStringEncoding by Encoder(true), | ||||
|     IMemSizer by NormalMemSizer(Mflpt5.FLOAT_MEM_SIZE) { | ||||
|  | ||||
|     override val name = NAME | ||||
|     override val defaultEncoding = Encoding.PETSCII | ||||
|     override val libraryPath = null | ||||
|     override val customLauncher = emptyList<String>() | ||||
|     override val additionalAssemblerOptions = emptyList<String>() | ||||
|     override val defaultOutputType = OutputType.PRG | ||||
|  | ||||
|     companion object { | ||||
|         const val NAME = "c64" | ||||
|  | ||||
|         fun viceMonListName(baseFilename: String) = "$baseFilename.vice-mon-list" | ||||
|     } | ||||
|  | ||||
|  | ||||
|     override val cpu = CpuType.CPU6502 | ||||
|  | ||||
|     override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE | ||||
|     override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE | ||||
|     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_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_END = 0xcfdfu | ||||
|     override val BSSGOLDENRAM_START = 0u        // no golden ram on C64 | ||||
|     override val BSSGOLDENRAM_END = 0u | ||||
|  | ||||
|     override lateinit var zeropage: Zeropage | ||||
|     override lateinit var golden: GoldenRam | ||||
|  | ||||
|     override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm() | ||||
|  | ||||
|     override fun convertFloatToBytes(num: Double): List<UByte> { | ||||
|         val m5 = Mflpt5.fromNumber(num) | ||||
|         return listOf(m5.b0, m5.b1, m5.b2, m5.b3, m5.b4) | ||||
|     } | ||||
|  | ||||
|     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, quiet: Boolean) { | ||||
|         if(selectedEmulator!=1) { | ||||
|             System.err.println("The c64 target only supports the main emulator (Vice).") | ||||
|             return | ||||
|         } | ||||
|  | ||||
|         for(emulator in listOf("x64sc", "x64")) { | ||||
|             if(!quiet) | ||||
|                 println("\nStarting C-64 emulator $emulator...") | ||||
|  | ||||
|             val viceMonlist = viceMonListName(programNameWithPath.toString()) | ||||
|             val cmdline = listOf(emulator, "-silent", "-moncommands", viceMonlist, | ||||
|                 "-autostartprgmode", "1", "-autostart-warp", "-autostart", "${programNameWithPath}.prg") | ||||
|             val processb = ProcessBuilder(cmdline) | ||||
|             if(!quiet) | ||||
|                 processb.inheritIO() | ||||
|             val process: Process | ||||
|             try { | ||||
|                 process=processb.start() | ||||
|             } catch(_: IOException) { | ||||
|                 continue  // try the next emulator executable | ||||
|             } | ||||
|             process.waitFor() | ||||
|             break | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     override fun isIOAddress(address: UInt): Boolean = address==0u || address==1u || address in 0xd000u..0xdfffu | ||||
|  | ||||
|     override fun initializeMemoryAreas(compilerOptions: CompilationOptions) { | ||||
|         zeropage = C64Zeropage(compilerOptions) | ||||
|         golden = GoldenRam(compilerOptions, UIntRange.EMPTY) | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
| val CompilationTargets = listOf( | ||||
|     C64Target.NAME, | ||||
|     C128Target.NAME, | ||||
|     Cx16Target.NAME, | ||||
|     PETTarget.NAME, | ||||
|     VMTarget.NAME | ||||
| ) | ||||
|  | ||||
| fun getCompilationTargetByName(name: String) = when(name.lowercase()) { | ||||
|     C64Target.NAME -> C64Target() | ||||
|     C128Target.NAME -> C128Target() | ||||
|     Cx16Target.NAME -> Cx16Target() | ||||
|     PETTarget.NAME -> PETTarget() | ||||
|     VMTarget.NAME -> VMTarget() | ||||
|     else -> throw IllegalArgumentException("invalid compilation target") | ||||
| } | ||||
							
								
								
									
										173
									
								
								codeCore/src/prog8/code/target/ConfigFileTarget.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										173
									
								
								codeCore/src/prog8/code/target/ConfigFileTarget.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,173 @@ | ||||
| package prog8.code.target | ||||
|  | ||||
| import prog8.code.core.* | ||||
| import prog8.code.source.ImportFileSystem.expandTilde | ||||
| import prog8.code.target.encodings.Encoder | ||||
| import prog8.code.target.zp.ConfigurableZeropage | ||||
| import java.io.IOException | ||||
| import java.nio.file.Path | ||||
| import java.util.* | ||||
| import kotlin.io.path.Path | ||||
| import kotlin.io.path.inputStream | ||||
| import kotlin.io.path.isDirectory | ||||
| import kotlin.io.path.nameWithoutExtension | ||||
|  | ||||
|  | ||||
| class ConfigFileTarget( | ||||
|     override val name: String, | ||||
|     override val defaultEncoding: Encoding, | ||||
|     override val cpu: CpuType, | ||||
|     override val PROGRAM_LOAD_ADDRESS: UInt, | ||||
|     override val PROGRAM_MEMTOP_ADDRESS: UInt, | ||||
|     override val STARTUP_CODE_RESERVED_SIZE: UInt, | ||||
|     override val BSSHIGHRAM_START: UInt, | ||||
|     override val BSSHIGHRAM_END: UInt, | ||||
|     override val BSSGOLDENRAM_START: UInt, | ||||
|     override val BSSGOLDENRAM_END: UInt, | ||||
|     override val defaultOutputType: OutputType, | ||||
|     override val libraryPath: Path, | ||||
|     override val customLauncher: List<String>, | ||||
|     override val additionalAssemblerOptions: List<String>, | ||||
|     val ioAddresses: List<UIntRange>, | ||||
|     val zpScratchB1: UInt, | ||||
|     val zpScratchReg: UInt, | ||||
|     val zpScratchW1: UInt, | ||||
|     val zpScratchW2: UInt, | ||||
|     val zpScratchPtr: UInt, | ||||
|     val virtualregistersStart: UInt, | ||||
|     val zpFullsafe: List<UIntRange>, | ||||
|     val zpKernalsafe: List<UIntRange>, | ||||
|     val zpBasicsafe: List<UIntRange> | ||||
| ): ICompilationTarget, IStringEncoding by Encoder(true), IMemSizer by NormalMemSizer(8) { | ||||
|  | ||||
|     companion object { | ||||
|  | ||||
|         private fun Properties.getString(property: String): String { | ||||
|             val value = this.getProperty(property, null) | ||||
|             if(value!=null) | ||||
|                 return value | ||||
|             throw NoSuchElementException("string property '$property' not found in config file") | ||||
|         } | ||||
|  | ||||
|         private fun Properties.getInteger(property: String): UInt { | ||||
|             val value = this.getProperty(property, null) | ||||
|             if(value!=null) return parseInt(value) | ||||
|             throw NoSuchElementException("integer property '$property' not found in config file") | ||||
|         } | ||||
|  | ||||
|         private fun parseInt(value: String): UInt { | ||||
|             if(value.startsWith("0x")) | ||||
|                 return value.drop(2).toUInt(16) | ||||
|             if(value.startsWith("$")) | ||||
|                 return value.drop(1).toUInt(16) | ||||
|             if(value.startsWith("%")) | ||||
|                 return value.drop(1).toUInt(2) | ||||
|             return value.toUInt() | ||||
|         } | ||||
|  | ||||
|         private fun parseAddressRanges(key: String, props: Properties): List<UIntRange> { | ||||
|             val rangesStr = props.getString(key) | ||||
|             if(rangesStr.isBlank()) | ||||
|                 return emptyList() | ||||
|             val result = mutableListOf<UIntRange>() | ||||
|             val ranges = rangesStr.split(",").map { it.trim() } | ||||
|             for(r in ranges) { | ||||
|                 if ('-' in r) { | ||||
|                     val (fromStr, toStr) = r.split("-") | ||||
|                     val from = parseInt(fromStr.trim()) | ||||
|                     val to = parseInt(toStr.trim()) | ||||
|                     result.add(from..to) | ||||
|                 } else { | ||||
|                     val address = parseInt(r) | ||||
|                     result.add(address..address) | ||||
|                 } | ||||
|             } | ||||
|             return result | ||||
|         } | ||||
|  | ||||
|         fun fromConfigFile(configfile: Path): ConfigFileTarget { | ||||
|             val props = Properties() | ||||
|             props.load(configfile.inputStream()) | ||||
|  | ||||
|             val cpuString = props.getString("cpu").uppercase() | ||||
|             val cpuType = try { | ||||
|                 CpuType.valueOf(cpuString) | ||||
|             } catch (_: IllegalArgumentException) { | ||||
|                 CpuType.valueOf("CPU$cpuString") | ||||
|             } | ||||
|             val ioAddresses = parseAddressRanges("io_regions", props) | ||||
|             val zpFullsafe = parseAddressRanges("zp_fullsafe", props) | ||||
|             val zpKernalsafe = parseAddressRanges("zp_kernalsafe", props) | ||||
|             val zpBasicsafe = parseAddressRanges("zp_basicsafe", props) | ||||
|  | ||||
|             val libraryPath = expandTilde(Path(props.getString("library"))) | ||||
|             if(!libraryPath.isDirectory()) | ||||
|                 throw IOException("invalid library path: $libraryPath") | ||||
|  | ||||
|             val customLauncherStr = props.getProperty("custom_launcher_code", null) | ||||
|             val customLauncher = | ||||
|                 if(customLauncherStr?.isNotBlank()==true) | ||||
|                     (customLauncherStr+"\n").lines().map { it.trimEnd() } | ||||
|                 else emptyList() | ||||
|             val assemblerOptionsStr = props.getProperty("assembler_options", "").trim() | ||||
|             val outputTypeString = props.getProperty("output_type", "PRG") | ||||
|             val defaultOutputType = OutputType.valueOf(outputTypeString.uppercase()) | ||||
|  | ||||
|             return ConfigFileTarget( | ||||
|                 configfile.nameWithoutExtension, | ||||
|                 Encoding.entries.first { it.prefix==props.getString("encoding") }, | ||||
|                 cpuType, | ||||
|                 props.getInteger("load_address"), | ||||
|                 props.getInteger("memtop"), | ||||
|                 0u,         // used only in a very specific error condition check in a certain scenario... | ||||
|                 props.getInteger("bss_highram_start"), | ||||
|                 props.getInteger("bss_highram_end"), | ||||
|                 props.getInteger("bss_goldenram_start"), | ||||
|                 props.getInteger("bss_goldenram_end"), | ||||
|                 defaultOutputType, | ||||
|                 libraryPath, | ||||
|                 customLauncher, | ||||
|                 if(assemblerOptionsStr=="") emptyList() else assemblerOptionsStr.split(" "), | ||||
|                 ioAddresses, | ||||
|                 props.getInteger("zp_scratch_b1"), | ||||
|                 props.getInteger("zp_scratch_reg"), | ||||
|                 props.getInteger("zp_scratch_w1"), | ||||
|                 props.getInteger("zp_scratch_w2"), | ||||
|                 props.getInteger("zp_scratch_ptr"), | ||||
|                 props.getInteger("virtual_registers"), | ||||
|                 zpFullsafe, | ||||
|                 zpKernalsafe, | ||||
|                 zpBasicsafe, | ||||
|             ) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // TODO floats are not yet supported here, just enter some values | ||||
|     override val FLOAT_MAX_POSITIVE = 9.999999999e97 | ||||
|     override val FLOAT_MAX_NEGATIVE = -9.999999999e97 | ||||
|     override val FLOAT_MEM_SIZE = 8 | ||||
|  | ||||
|     override lateinit var zeropage: Zeropage | ||||
|     override lateinit var golden: GoldenRam     // TODO this is not yet used | ||||
|  | ||||
|     override fun getFloatAsmBytes(num: Number) = TODO("floats") | ||||
|     override fun convertFloatToBytes(num: Double): List<UByte> = TODO("floats") | ||||
|     override fun convertBytesToFloat(bytes: List<UByte>): Double = TODO("floats") | ||||
|     override fun launchEmulator(selectedEmulator: Int, programNameWithPath: Path, quiet: Boolean) { | ||||
|         throw IllegalArgumentException("Custom compiler target cannot automatically launch an emulator. Do this manually.") | ||||
|     } | ||||
|  | ||||
|     override fun isIOAddress(address: UInt): Boolean = ioAddresses.any { address in it } | ||||
|  | ||||
|     override fun initializeMemoryAreas(compilerOptions: CompilationOptions) { | ||||
|         zeropage = ConfigurableZeropage( | ||||
|             zpScratchB1, zpScratchReg, zpScratchW1, zpScratchW2, zpScratchPtr, | ||||
|             virtualregistersStart, | ||||
|             zpBasicsafe, | ||||
|             zpKernalsafe, | ||||
|             zpFullsafe, | ||||
|             compilerOptions | ||||
|         ) | ||||
|         // note: there's no golden ram yet | ||||
|     } | ||||
| } | ||||
							
								
								
									
										94
									
								
								codeCore/src/prog8/code/target/Cx16Target.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										94
									
								
								codeCore/src/prog8/code/target/Cx16Target.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,94 @@ | ||||
| package prog8.code.target | ||||
|  | ||||
| import prog8.code.core.* | ||||
| import prog8.code.target.encodings.Encoder | ||||
| import prog8.code.target.zp.CX16Zeropage | ||||
| import java.nio.file.Path | ||||
|  | ||||
|  | ||||
| class Cx16Target: ICompilationTarget, | ||||
|     IStringEncoding by Encoder(true), | ||||
|     IMemSizer by NormalMemSizer(Mflpt5.FLOAT_MEM_SIZE) { | ||||
|  | ||||
|     override val name = NAME | ||||
|     override val defaultEncoding = Encoding.PETSCII | ||||
|     override val libraryPath = null | ||||
|     override val customLauncher = emptyList<String>() | ||||
|     override val additionalAssemblerOptions = emptyList<String>() | ||||
|     override val defaultOutputType = OutputType.PRG | ||||
|  | ||||
|     companion object { | ||||
|         const val NAME = "cx16" | ||||
|     } | ||||
|  | ||||
|  | ||||
|     override val cpu = CpuType.CPU65C02 | ||||
|  | ||||
|     override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE | ||||
|     override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE | ||||
|     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_MEMTOP_ADDRESS = 0x9f00u | ||||
|  | ||||
|     override val BSSHIGHRAM_START = 0xa000u     // hiram bank 1, 8Kb, assumed to be active | ||||
|     override val BSSHIGHRAM_END = 0xbfffu       // Rom starts at $c000 | ||||
|     override val BSSGOLDENRAM_START = 0x0400u | ||||
|     override val BSSGOLDENRAM_END = 0x07ffu | ||||
|  | ||||
|     override lateinit var zeropage: Zeropage | ||||
|     override lateinit var golden: GoldenRam | ||||
|  | ||||
|     override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm() | ||||
|  | ||||
|     override fun convertFloatToBytes(num: Double): List<UByte> { | ||||
|         val m5 = Mflpt5.fromNumber(num) | ||||
|         return listOf(m5.b0, m5.b1, m5.b2, m5.b3, m5.b4) | ||||
|     } | ||||
|  | ||||
|     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, quiet: Boolean) { | ||||
|         val emulator: String | ||||
|         val extraArgs: List<String> | ||||
|  | ||||
|         when(selectedEmulator) { | ||||
|             1 -> { | ||||
|                 emulator = "x16emu" | ||||
|                 extraArgs = listOf("-debug") | ||||
|             } | ||||
|             2 -> { | ||||
|                 emulator = "box16" | ||||
|                 extraArgs = listOf("-sym", C64Target.viceMonListName(programNameWithPath.toString())) | ||||
|             } | ||||
|             else -> { | ||||
|                 System.err.println("Cx16 target only supports x16emu and box16 emulators.") | ||||
|                 return | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if(!quiet) | ||||
|             println("\nStarting Commander X16 emulator $emulator...") | ||||
|  | ||||
|         val cmdline = listOf(emulator, "-scale", "2", "-rtc", "-run", "-prg", "${programNameWithPath}.prg") + extraArgs | ||||
|         val processb = ProcessBuilder(cmdline) | ||||
|         if(!quiet) | ||||
|             processb.inheritIO() | ||||
|         processb.environment()["PULSE_LATENCY_MSEC"] = "10" | ||||
|         val process: Process = processb.start() | ||||
|         process.waitFor() | ||||
|     } | ||||
|  | ||||
|     override fun isIOAddress(address: UInt): Boolean = address==0u || address==1u || address in 0x9f00u..0x9fffu | ||||
|  | ||||
|     override fun initializeMemoryAreas(compilerOptions: CompilationOptions) { | ||||
|         zeropage = CX16Zeropage(compilerOptions) | ||||
|         golden = GoldenRam(compilerOptions, BSSGOLDENRAM_START..BSSGOLDENRAM_END) | ||||
|     } | ||||
|  | ||||
|  | ||||
| } | ||||
| @@ -1,12 +1,10 @@ | ||||
| package prog8.compiler.target.cbm | ||||
| package prog8.code.target | ||||
| 
 | ||||
| import prog8.compilerinterface.IMachineFloat | ||||
| import prog8.compilerinterface.InternalCompilerException | ||||
| import prog8.code.core.InternalCompilerException | ||||
| import kotlin.math.absoluteValue | ||||
| import kotlin.math.pow | ||||
| 
 | ||||
| 
 | ||||
| data class Mflpt5(val b0: UByte, val b1: UByte, val b2: UByte, val b3: UByte, val b4: UByte): IMachineFloat { | ||||
| data class Mflpt5(val b0: UByte, val b1: UByte, val b2: UByte, val b3: UByte, val b4: UByte) { | ||||
| 
 | ||||
|     companion object { | ||||
|         const val FLOAT_MAX_POSITIVE = 1.7014118345e+38         // bytes: 255,127,255,255,255 | ||||
| @@ -20,7 +18,7 @@ data class Mflpt5(val b0: UByte, val b1: UByte, val b2: UByte, val b3: UByte, va | ||||
|             // and https://en.wikipedia.org/wiki/IEEE_754-1985 | ||||
| 
 | ||||
|             val flt = num.toDouble() | ||||
|             if (flt < FLOAT_MAX_NEGATIVE || flt > FLOAT_MAX_POSITIVE) | ||||
|             if (flt !in FLOAT_MAX_NEGATIVE..FLOAT_MAX_POSITIVE) | ||||
|                 throw InternalCompilerException("floating point number out of 5-byte mflpt range: $this") | ||||
|             if (flt == 0.0) | ||||
|                 return zero | ||||
| @@ -58,7 +56,7 @@ data class Mflpt5(val b0: UByte, val b1: UByte, val b2: UByte, val b3: UByte, va | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     override fun toDouble(): Double { | ||||
|     fun toDouble(): Double { | ||||
|         if (this == zero) return 0.0 | ||||
|         val exp = b0.toInt() - 128 | ||||
|         val sign = (b1.toInt() and 0x80) > 0 | ||||
| @@ -67,7 +65,7 @@ data class Mflpt5(val b0: UByte, val b1: UByte, val b2: UByte, val b3: UByte, va | ||||
|         return if (sign) -result else result | ||||
|     } | ||||
| 
 | ||||
|     override fun makeFloatFillAsm(): String { | ||||
|     fun makeFloatFillAsm(): String { | ||||
|         val b0 = "$" + b0.toString(16).padStart(2, '0') | ||||
|         val b1 = "$" + b1.toString(16).padStart(2, '0') | ||||
|         val b2 = "$" + b2.toString(16).padStart(2, '0') | ||||
							
								
								
									
										38
									
								
								codeCore/src/prog8/code/target/NormalMemSizer.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								codeCore/src/prog8/code/target/NormalMemSizer.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| package prog8.code.target | ||||
|  | ||||
| import prog8.code.core.BaseDataType | ||||
| import prog8.code.core.DataType | ||||
| import prog8.code.core.IMemSizer | ||||
|  | ||||
| internal class NormalMemSizer(val floatsize: Int): IMemSizer { | ||||
|  | ||||
|     override fun memorySize(dt: DataType, numElements: Int?): Int { | ||||
|         if(dt.isPointerArray) | ||||
|             return 2 * numElements!!        // array of pointers is just array of uwords | ||||
|         else 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 * floatsize | ||||
|                 BaseDataType.UNDEFINED -> throw IllegalArgumentException("undefined has no memory size") | ||||
|                 else -> throw IllegalArgumentException("invalid sub type") | ||||
|             } | ||||
|         } | ||||
|         else if (dt.isString) { | ||||
|             return numElements        // treat it as the size of the given string with the length | ||||
|                 ?: 2    // treat it as the size to store a string pointer | ||||
|         } | ||||
|  | ||||
|         return when { | ||||
|             dt.isByteOrBool -> 1 * (numElements ?: 1) | ||||
|             dt.isFloat -> floatsize * (numElements ?: 1) | ||||
|             dt.isLong -> 4 * (numElements ?: 1) | ||||
|             dt.isPointer -> 2  // pointer is just a uword | ||||
|             dt.isStructInstance -> dt.subType!!.memsize(this) | ||||
|             dt.isUndefined -> throw IllegalArgumentException("undefined has no memory size") | ||||
|             else -> 2 * (numElements ?: 1) | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										80
									
								
								codeCore/src/prog8/code/target/PETTarget.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										80
									
								
								codeCore/src/prog8/code/target/PETTarget.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,80 @@ | ||||
| package prog8.code.target | ||||
|  | ||||
| import prog8.code.core.* | ||||
| import prog8.code.target.encodings.Encoder | ||||
| import prog8.code.target.zp.PETZeropage | ||||
| import java.nio.file.Path | ||||
|  | ||||
|  | ||||
| class PETTarget: ICompilationTarget, | ||||
|     IStringEncoding by Encoder(true), | ||||
|     IMemSizer by NormalMemSizer(Mflpt5.FLOAT_MEM_SIZE) { | ||||
|  | ||||
|     override val name = NAME | ||||
|     override val defaultEncoding = Encoding.PETSCII | ||||
|     override val libraryPath = null | ||||
|     override val customLauncher = emptyList<String>() | ||||
|     override val additionalAssemblerOptions = emptyList<String>() | ||||
|     override val defaultOutputType = OutputType.PRG | ||||
|  | ||||
|     companion object { | ||||
|         const val NAME = "pet32" | ||||
|     } | ||||
|  | ||||
|     override val cpu = CpuType.CPU6502 | ||||
|  | ||||
|     override val FLOAT_MAX_POSITIVE = Mflpt5.FLOAT_MAX_POSITIVE | ||||
|     override val FLOAT_MAX_NEGATIVE = Mflpt5.FLOAT_MAX_NEGATIVE | ||||
|     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_MEMTOP_ADDRESS = 0x8000u | ||||
|  | ||||
|     override val BSSHIGHRAM_START = 0u | ||||
|     override val BSSHIGHRAM_END = 0u | ||||
|     override val BSSGOLDENRAM_START = 0u | ||||
|     override val BSSGOLDENRAM_END = 0u | ||||
|  | ||||
|     override lateinit var zeropage: Zeropage | ||||
|     override lateinit var golden: GoldenRam | ||||
|  | ||||
|     override fun getFloatAsmBytes(num: Number) = Mflpt5.fromNumber(num).makeFloatFillAsm() | ||||
|  | ||||
|     override fun convertFloatToBytes(num: Double): List<UByte> { | ||||
|         val m5 = Mflpt5.fromNumber(num) | ||||
|         return listOf(m5.b0, m5.b1, m5.b2, m5.b3, m5.b4) | ||||
|     } | ||||
|  | ||||
|     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, quiet: Boolean) { | ||||
|         if(selectedEmulator!=1) { | ||||
|             System.err.println("The pet target only supports the main emulator (Vice).") | ||||
|             return | ||||
|         } | ||||
|  | ||||
|         if(!quiet) | ||||
|             println("\nStarting PET emulator...") | ||||
|  | ||||
|         val viceMonlist = C64Target.viceMonListName(programNameWithPath.toString()) | ||||
|         val cmdline = listOf("xpet", "-model", "4032", "-ramsize", "32", "-videosize", "40", "-silent", "-moncommands", viceMonlist, | ||||
|             "-autostartprgmode", "1", "-autostart-warp", "-autostart", "${programNameWithPath}.prg") | ||||
|         val processb = ProcessBuilder(cmdline) | ||||
|         if(!quiet) | ||||
|             processb.inheritIO() | ||||
|         val process=processb.start() | ||||
|         process.waitFor() | ||||
|     } | ||||
|  | ||||
|     override fun isIOAddress(address: UInt): Boolean = address in 0xe800u..0xe8ffu | ||||
|  | ||||
|     override fun initializeMemoryAreas(compilerOptions: CompilationOptions) { | ||||
|         zeropage = PETZeropage(compilerOptions) | ||||
|         golden = GoldenRam(compilerOptions, UIntRange.EMPTY) | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										109
									
								
								codeCore/src/prog8/code/target/VMTarget.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										109
									
								
								codeCore/src/prog8/code/target/VMTarget.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,109 @@ | ||||
| package prog8.code.target | ||||
|  | ||||
| import prog8.code.core.* | ||||
| import prog8.code.target.encodings.Encoder | ||||
| import java.nio.file.Path | ||||
| import kotlin.io.path.extension | ||||
| import kotlin.io.path.isReadable | ||||
| import kotlin.io.path.name | ||||
| import kotlin.io.path.readText | ||||
|  | ||||
| class VMTarget: ICompilationTarget, | ||||
|     IStringEncoding by Encoder(false), | ||||
|     IMemSizer by NormalMemSizer(FLOAT_MEM_SIZE) { | ||||
|  | ||||
|     override val name = NAME | ||||
|     override val defaultEncoding = Encoding.ISO | ||||
|     override val libraryPath = null | ||||
|     override val customLauncher = emptyList<String>() | ||||
|     override val additionalAssemblerOptions = emptyList<String>() | ||||
|     override val defaultOutputType = OutputType.PRG | ||||
|  | ||||
|     companion object { | ||||
|         const val NAME = "virtual" | ||||
|         const val FLOAT_MEM_SIZE = 8             // 64-bits double | ||||
|     } | ||||
|  | ||||
|     override val cpu = CpuType.VIRTUAL | ||||
|  | ||||
|     override val FLOAT_MAX_POSITIVE = Double.MAX_VALUE | ||||
|     override val FLOAT_MAX_NEGATIVE = -Double.MAX_VALUE | ||||
|     override val FLOAT_MEM_SIZE = VMTarget.FLOAT_MEM_SIZE | ||||
|     override val STARTUP_CODE_RESERVED_SIZE = 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_END = 0u            // not actually used | ||||
|     override val BSSGOLDENRAM_START = 0u        // not actually used | ||||
|     override val BSSGOLDENRAM_END = 0u          // not actually used | ||||
|     override lateinit var zeropage: Zeropage    // not actually used | ||||
|     override lateinit var golden: GoldenRam     // not actually used | ||||
|  | ||||
|     override fun getFloatAsmBytes(num: Number): String { | ||||
|         // little endian binary representation | ||||
|         val bits = num.toDouble().toBits().toULong() | ||||
|         val hexStr = bits.toString(16).padStart(16, '0') | ||||
|         val parts = hexStr.chunked(2).map { "$$it" } | ||||
|         return parts.joinToString(", ") | ||||
|     } | ||||
|  | ||||
|     override fun convertFloatToBytes(num: Double): List<UByte> { | ||||
|         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, quiet: Boolean) { | ||||
|         if(!quiet) | ||||
|             println("\nStarting Virtual Machine...") | ||||
|  | ||||
|         // to not have external module dependencies in our own module, we launch the virtual machine via reflection | ||||
|         val vm = Class.forName("prog8.vm.VmRunner").getDeclaredConstructor().newInstance() as IVirtualMachineRunner | ||||
|         val withExt = if(programNameWithPath.extension=="p8ir") programNameWithPath else programNameWithPath.resolveSibling("${programNameWithPath.name}.p8ir") | ||||
|         if(withExt.isReadable()) | ||||
|             vm.runProgram(withExt.readText(), quiet) | ||||
|         else | ||||
|             throw java.nio.file.NoSuchFileException(withExt.name, null, "not a .p8ir file") | ||||
|     } | ||||
|  | ||||
|     override fun isIOAddress(address: UInt): Boolean = false | ||||
|  | ||||
|     override fun initializeMemoryAreas(compilerOptions: CompilationOptions) { | ||||
|         zeropage = VirtualZeropage(compilerOptions) | ||||
|         golden = GoldenRam(compilerOptions, UIntRange.EMPTY) | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| interface IVirtualMachineRunner { | ||||
|     fun runProgram(irSource: String, quiet: Boolean) | ||||
| } | ||||
|  | ||||
| private class VirtualZeropage(options: CompilationOptions): Zeropage(options) { | ||||
|     override val SCRATCH_B1: UInt | ||||
|         get() = throw IllegalStateException("virtual shouldn't use this zeropage variable") | ||||
|     override val SCRATCH_REG: UInt | ||||
|         get() = throw IllegalStateException("virtual shouldn't use this zeropage variable") | ||||
|     override val SCRATCH_W1: UInt | ||||
|         get() = throw IllegalStateException("virtual shouldn't use this zeropage variable") | ||||
|     override val SCRATCH_W2: UInt | ||||
|         get() = throw IllegalStateException("virtual shouldn't use this zeropage variable") | ||||
|     override val SCRATCH_PTR: UInt | ||||
|         get() = throw IllegalStateException("virtual shouldn't use this zeropage variable") | ||||
| } | ||||
							
								
								
									
										215
									
								
								codeCore/src/prog8/code/target/encodings/AtasciiEncoding.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										215
									
								
								codeCore/src/prog8/code/target/encodings/AtasciiEncoding.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,215 @@ | ||||
| package prog8.code.target.encodings | ||||
|  | ||||
| import com.github.michaelbull.result.Ok | ||||
| import com.github.michaelbull.result.Result | ||||
| import java.io.CharConversionException | ||||
|  | ||||
| object AtasciiEncoding { | ||||
|  | ||||
|     private val decodeTable: CharArray = charArrayOf( | ||||
|         // $00 | ||||
|         '♥', | ||||
|         '├', | ||||
|         '\uf130',    //  🮇    0x02 -> RIGHT ONE QUARTER BLOCK (CUS) | ||||
|         '┘', | ||||
|         '┤', | ||||
|         '┐', | ||||
|         '╱', | ||||
|         '╲', | ||||
|         '◢', | ||||
|         '▗', | ||||
|         '◣', | ||||
|         '▝', | ||||
|         '▘', | ||||
|         '\uf132',    //  🮂    0x1d -> UPPER ONE QUARTER BLOCK (CUS) | ||||
|         '▂', | ||||
|         '▖', | ||||
|  | ||||
|         // $10 | ||||
|         '♣', | ||||
|         '┌', | ||||
|         '─', | ||||
|         '┼', | ||||
|         '•', | ||||
|         '▄', | ||||
|         '▎', | ||||
|         '┬', | ||||
|         '┴', | ||||
|         '▌', | ||||
|         '└', | ||||
|         '\u001b',           // $1b = escape | ||||
|         '\ufffe',           // UNDEFINED CHAR.   $1c = cursor up | ||||
|         '\ufffe',           // UNDEFINED CHAR.   $1d = cursor down | ||||
|         '\ufffe',           // UNDEFINED CHAR.   $1e = cursor left | ||||
|         '\ufffe',           // UNDEFINED CHAR.   $1f = cursor right | ||||
|  | ||||
|         // $20 | ||||
|         ' ', | ||||
|         '!', | ||||
|         '"', | ||||
|         '#', | ||||
|         '$', | ||||
|         '%', | ||||
|         '&', | ||||
|         '\'', | ||||
|         '(', | ||||
|         ')', | ||||
|         '*', | ||||
|         '+', | ||||
|         ',', | ||||
|         '-', | ||||
|         '.', | ||||
|         '/', | ||||
|  | ||||
|         // $30 | ||||
|         '0', | ||||
|         '1', | ||||
|         '2', | ||||
|         '3', | ||||
|         '4', | ||||
|         '5', | ||||
|         '6', | ||||
|         '7', | ||||
|         '8', | ||||
|         '9', | ||||
|         ':', | ||||
|         ';', | ||||
|         '<', | ||||
|         '=', | ||||
|         '>', | ||||
|         '?', | ||||
|  | ||||
|         // $40 | ||||
|         '@', | ||||
|         'A', | ||||
|         'B', | ||||
|         'C', | ||||
|         'D', | ||||
|         'E', | ||||
|         'F', | ||||
|         'G', | ||||
|         'H', | ||||
|         'I', | ||||
|         'J', | ||||
|         'K', | ||||
|         'L', | ||||
|         'M', | ||||
|         'N', | ||||
|         'O', | ||||
|  | ||||
|         // $50 | ||||
|         'P', | ||||
|         'Q', | ||||
|         'R', | ||||
|         'S', | ||||
|         'T', | ||||
|         'U', | ||||
|         'V', | ||||
|         'W', | ||||
|         'X', | ||||
|         'Y', | ||||
|         'Z', | ||||
|         '[', | ||||
|         '\\', | ||||
|         ']', | ||||
|         '^', | ||||
|         '_', | ||||
|  | ||||
|         // $60 | ||||
|         '♦', | ||||
|         'a', | ||||
|         'b', | ||||
|         'c', | ||||
|         'd', | ||||
|         'e', | ||||
|         'f', | ||||
|         'g', | ||||
|         'h', | ||||
|         'i', | ||||
|         'j', | ||||
|         'k', | ||||
|         'l', | ||||
|         'm', | ||||
|         'n', | ||||
|         'o', | ||||
|  | ||||
|         // $70 | ||||
|         'p', | ||||
|         'q', | ||||
|         'r', | ||||
|         's', | ||||
|         't', | ||||
|         'u', | ||||
|         'v', | ||||
|         'w', | ||||
|         'x', | ||||
|         'y', | ||||
|         'z', | ||||
|         '♠', | ||||
|         '|', | ||||
|         '\u000c',    //       $7d -> FORM FEED (CLEAR SCREEN) | ||||
|         '\u0008',    //       $7e -> BACKSPACE | ||||
|         '\u0009',    //       $7f -> TAB | ||||
|  | ||||
|         // $80-$ff are reversed video characters + various special characters. | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         // $90 | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         '\ufffe', | ||||
|         '\ufffe', | ||||
|         '\ufffe', | ||||
|         '\n',               // $9b -> EOL/RETURN | ||||
|         '\ufffe',           // UNDEFINED   $9c = DELETE LINE | ||||
|         '\ufffe',           // UNDEFINED   $9d = INSERT LINE | ||||
|         '\ufffe',           // UNDEFINED   $9e = CLEAR TAB STOP | ||||
|         '\ufffe',           // UNDEFINED   $9f = SET TAB STOP | ||||
|         // $a0 | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         // $b0 | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         // $c0 | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         // $d0 | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         // $e0 | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         // $f0 | ||||
|         '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', '\ufffe', | ||||
|         '\ufffe', | ||||
|         '\ufffe', | ||||
|         '\ufffe', | ||||
|         '\ufffe', | ||||
|         '\ufffe', | ||||
|         '\u0007',           // $fd = bell/beep | ||||
|         '\u007f',           // $fe = DELETE | ||||
|         '\ufffe'            // UNDEFINED     $ff = INSERT | ||||
|     ) | ||||
|  | ||||
|     private val encodeTable = decodeTable.withIndex().associate{it.value to it.index} | ||||
|  | ||||
|  | ||||
|     fun encode(str: String): Result<List<UByte>, CharConversionException> { | ||||
|         val mapped = str.map { chr -> | ||||
|             when (chr) { | ||||
|                 '\r' -> 0x9bu | ||||
|                 '\u0000' -> 0u | ||||
|                 in '\u8000'..'\u80ff' -> { | ||||
|                     // special case: take the lower 8 bit hex value directly | ||||
|                     (chr.code - 0x8000).toUByte() | ||||
|                 } | ||||
|                 else -> encodeTable.getValue(chr).toUByte() | ||||
|             } | ||||
|         } | ||||
|         return Ok(mapped) | ||||
|     } | ||||
|  | ||||
|     fun decode(bytes: Iterable<UByte>): Result<String, CharConversionException> { | ||||
|         return Ok(bytes.map { decodeTable[it.toInt()] }.joinToString("")) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										327
									
								
								codeCore/src/prog8/code/target/encodings/C64osEncoding.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										327
									
								
								codeCore/src/prog8/code/target/encodings/C64osEncoding.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,327 @@ | ||||
| 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 | ||||
|  | ||||
| object C64osEncoding { | ||||
|  | ||||
|     // decoding:  from C64 OS Screencodes (0-255) to unicode | ||||
|     // character table from: | ||||
|     // https://www.c64os.com/c64os/usersguide/appendices#charactersets | ||||
|  | ||||
|     private val decodingC64os = charArrayOf( | ||||
|         '@'     ,    //  @    0x00 -> COMMERCIAL AT | ||||
|         'a'     ,    //  a    0x01 -> LATIN SMALL LETTER A | ||||
|         'b'     ,    //  b    0x02 -> LATIN SMALL LETTER B | ||||
|         'c'     ,    //  c    0x03 -> LATIN SMALL LETTER C | ||||
|         'd'     ,    //  d    0x04 -> LATIN SMALL LETTER D | ||||
|         'e'     ,    //  e    0x05 -> LATIN SMALL LETTER E | ||||
|         'f'     ,    //  f    0x06 -> LATIN SMALL LETTER F | ||||
|         'g'     ,    //  g    0x07 -> LATIN SMALL LETTER G | ||||
|         'h'     ,    //  h    0x08 -> LATIN SMALL LETTER H | ||||
|         'i'     ,    //  i    0x09 -> LATIN SMALL LETTER I | ||||
|         'j'     ,    //  j    0x0A -> LATIN SMALL LETTER J | ||||
|         'k'     ,    //  k    0x0B -> LATIN SMALL LETTER K | ||||
|         'l'     ,    //  l    0x0C -> LATIN SMALL LETTER L | ||||
|         'm'     ,    //  m    0x0D -> LATIN SMALL LETTER M | ||||
|         'n'     ,    //  n    0x0E -> LATIN SMALL LETTER N | ||||
|         'o'     ,    //  o    0x0F -> LATIN SMALL LETTER O | ||||
|         'p'     ,    //  p    0x10 -> LATIN SMALL LETTER P | ||||
|         'q'     ,    //  q    0x11 -> LATIN SMALL LETTER Q | ||||
|         'r'     ,    //  r    0x12 -> LATIN SMALL LETTER R | ||||
|         's'     ,    //  s    0x13 -> LATIN SMALL LETTER S | ||||
|         't'     ,    //  t    0x14 -> LATIN SMALL LETTER T | ||||
|         'u'     ,    //  u    0x15 -> LATIN SMALL LETTER U | ||||
|         'v'     ,    //  v    0x16 -> LATIN SMALL LETTER V | ||||
|         'w'     ,    //  w    0x17 -> LATIN SMALL LETTER W | ||||
|         'x'     ,    //  x    0x18 -> LATIN SMALL LETTER X | ||||
|         'y'     ,    //  y    0x19 -> LATIN SMALL LETTER Y | ||||
|         'z'     ,    //  z    0x1A -> LATIN SMALL LETTER Z | ||||
|         '['     ,    //  [    0x1B -> LEFT SQUARE BRACKET | ||||
|         '\\'    ,    //  \    0x1C -> REVERSE SOLIDUS | ||||
|         ']'     ,    //  ]    0x1D -> RIGHT SQUARE BRACKET | ||||
|         '^'     ,    //  ^    0x1E -> CIRCUMFLEX | ||||
|         '_'     ,    //  _    0x1F -> UNDERSCORE | ||||
|         ' '     ,    //       0x20 -> SPACE | ||||
|         '!'     ,    //  !    0x21 -> EXCLAMATION MARK | ||||
|         '"'     ,    //  "    0x22 -> QUOTATION MARK | ||||
|         '#'     ,    //  #    0x23 -> NUMBER SIGN | ||||
|         '$'     ,    //  $    0x24 -> DOLLAR SIGN | ||||
|         '%'     ,    //  %    0x25 -> PERCENT SIGN | ||||
|         '&'     ,    //  &    0x26 -> AMPERSAND | ||||
|         '\''    ,    //  '    0x27 -> APOSTROPHE | ||||
|         '('     ,    //  (    0x28 -> LEFT PARENTHESIS | ||||
|         ')'     ,    //  )    0x29 -> RIGHT PARENTHESIS | ||||
|         '*'     ,    //  *    0x2A -> ASTERISK | ||||
|         '+'     ,    //  +    0x2B -> PLUS SIGN | ||||
|         ','     ,    //  ,    0x2C -> COMMA | ||||
|         '-'     ,    //  -    0x2D -> HYPHEN-MINUS | ||||
|         '.'     ,    //  .    0x2E -> FULL STOP | ||||
|         '/'     ,    //  /    0x2F -> SOLIDUS | ||||
|         '0'     ,    //  0    0x30 -> DIGIT ZERO | ||||
|         '1'     ,    //  1    0x31 -> DIGIT ONE | ||||
|         '2'     ,    //  2    0x32 -> DIGIT TWO | ||||
|         '3'     ,    //  3    0x33 -> DIGIT THREE | ||||
|         '4'     ,    //  4    0x34 -> DIGIT FOUR | ||||
|         '5'     ,    //  5    0x35 -> DIGIT FIVE | ||||
|         '6'     ,    //  6    0x36 -> DIGIT SIX | ||||
|         '7'     ,    //  7    0x37 -> DIGIT SEVEN | ||||
|         '8'     ,    //  8    0x38 -> DIGIT EIGHT | ||||
|         '9'     ,    //  9    0x39 -> DIGIT NINE | ||||
|         ':'     ,    //  :    0x3A -> COLON | ||||
|         ';'     ,    //  ;    0x3B -> SEMICOLON | ||||
|         '<'     ,    //  <    0x3C -> LESS-THAN SIGN | ||||
|         '='     ,    //  =    0x3D -> EQUALS SIGN | ||||
|         '>'     ,    //  >    0x3E -> GREATER-THAN SIGN | ||||
|         '?'     ,    //  ?    0x3F -> QUESTION MARK | ||||
|         '`'     ,    //  `    0x40 -> GRAVE ACCENT | ||||
|         'A'     ,    //  A    0x41 -> LATIN CAPITAL LETTER A | ||||
|         'B'     ,    //  B    0x42 -> LATIN CAPITAL LETTER B | ||||
|         'C'     ,    //  C    0x43 -> LATIN CAPITAL LETTER C | ||||
|         'D'     ,    //  D    0x44 -> LATIN CAPITAL LETTER D | ||||
|         'E'     ,    //  E    0x45 -> LATIN CAPITAL LETTER E | ||||
|         'F'     ,    //  F    0x46 -> LATIN CAPITAL LETTER F | ||||
|         'G'     ,    //  G    0x47 -> LATIN CAPITAL LETTER G | ||||
|         'H'     ,    //  H    0x48 -> LATIN CAPITAL LETTER H | ||||
|         'I'     ,    //  I    0x49 -> LATIN CAPITAL LETTER I | ||||
|         'J'     ,    //  J    0x4A -> LATIN CAPITAL LETTER J | ||||
|         'K'     ,    //  K    0x4B -> LATIN CAPITAL LETTER K | ||||
|         'L'     ,    //  L    0x4C -> LATIN CAPITAL LETTER L | ||||
|         'M'     ,    //  M    0x4D -> LATIN CAPITAL LETTER M | ||||
|         'N'     ,    //  N    0x4E -> LATIN CAPITAL LETTER N | ||||
|         'O'     ,    //  O    0x4F -> LATIN CAPITAL LETTER O | ||||
|         'P'     ,    //  P    0x50 -> LATIN CAPITAL LETTER P | ||||
|         'Q'     ,    //  Q    0x51 -> LATIN CAPITAL LETTER Q | ||||
|         'R'     ,    //  R    0x52 -> LATIN CAPITAL LETTER R | ||||
|         'S'     ,    //  S    0x53 -> LATIN CAPITAL LETTER S | ||||
|         'T'     ,    //  T    0x54 -> LATIN CAPITAL LETTER T | ||||
|         'U'     ,    //  U    0x55 -> LATIN CAPITAL LETTER U | ||||
|         'V'     ,    //  V    0x56 -> LATIN CAPITAL LETTER V | ||||
|         'W'     ,    //  W    0x57 -> LATIN CAPITAL LETTER W | ||||
|         'X'     ,    //  X    0x58 -> LATIN CAPITAL LETTER X | ||||
|         'Y'     ,    //  Y    0x59 -> LATIN CAPITAL LETTER Y | ||||
|         'Z'     ,    //  Z    0x5A -> LATIN CAPITAL LETTER Z | ||||
|         '{'     ,    //  {    0x5B -> LEFT BRACE | ||||
|         '|'     ,    //  |    0x5C -> VERTICAL BAR | ||||
|         '}'     ,    //  }    0x5D -> RIGHT BRACE | ||||
|         '~'     ,    //  ~    0x5E -> TILDE | ||||
|         '\ufffe',    //       0x5F -> RESERVED | ||||
|         '\u00a0',    //       0x60 -> NO-BREAK SPACE (TRANSPARENT) | ||||
|         '\ufffe',    //       0x61 -> COMMODORE SYMBOL | ||||
|         '\u2191',    //  ↑    0x62 -> UP ARROW | ||||
|         '\u2193',    //  ↓    0x63 -> DOWN ARROW | ||||
|         '\u2190',    //  ←    0x64 -> LEFT ARROW | ||||
|         '\u2192',    //  →    0x65 -> RIGHT ARROW | ||||
|         '\u231A',    //  ⌚   0x66 -> WATCH (ANALOG CLOCKFACE) | ||||
|         '\u21BB',    //  ↻    0x67 -> CYCLE ARROWS | ||||
|         '\u2026',    //  …    0x68 -> ELLIPSIS | ||||
|         '\u25a7',    //  ▧    0x69 -> DIAGNONAL STRIPES | ||||
|         '\u2610',    //  ☐    0x6A -> CHECKBOX UNCHECKED | ||||
|         '\u2611',    //  ☑    0x6B -> CHECKBOX CHECKED | ||||
|         '\ufffe',    //       0x6C -> RADIO BUTTON UNSELECTED | ||||
|         '\ufffe',    //       0x6D -> RADIO BUTTON SELECTED | ||||
|         '\ufffe',    //       0x6E -> UTILITY CLOSE BUTTON | ||||
|         '\ufffe',    //       0x6F -> UTILITY TITLE BAR | ||||
|         '\u00a9',    //  ©    0x70 -> COPYRIGHT | ||||
|         '\u2713',    //  ✓    0x71 -> CHECKMARK | ||||
|         '\u2261',    //  ≡    0x72 -> THREE HORIZONTAL STRIPES | ||||
|         '\ufffe',    //       0x73 -> TICK TRACK | ||||
|         '\ufffe',    //       0x74 -> TICK TRACK NUB | ||||
|         '\ufffe',    //       0x75 -> TAB CORNER | ||||
|         '\u2980',    //  ⦀    0x76 -> THREE VERTICAL STRIPES | ||||
|         '\ufffe',    //       0x77 -> CUSTOM 1 | ||||
|         '\ufffe',    //       0x78 -> CUSTOM 2 | ||||
|         '\ufffe',    //       0x79 -> CUSTOM 3 | ||||
|         '\ufffe',    //       0x7A -> CUSTOM 4 | ||||
|         '\ufffe',    //       0x7B -> CUSTOM 5 | ||||
|         '\ufffe',    //       0x7C -> CUSTOM 6 | ||||
|         '\ufffe',    //       0x7D -> CUSTOM 7 | ||||
|         '\ufffe',    //       0x7E -> CUSTOM 8 | ||||
|         '\ufffe',    //       0x7F -> CUSTOM 9 | ||||
|         '\ufffe',    //       0x80 -> REVERSED COMMERCIAL AT | ||||
|         '\ufffe',    //       0x81 -> REVERSED LATIN SMALL LETTER A | ||||
|         '\ufffe',    //       0x82 -> REVERSED LATIN SMALL LETTER B | ||||
|         '\ufffe',    //       0x83 -> REVERSED LATIN SMALL LETTER C | ||||
|         '\ufffe',    //       0x84 -> REVERSED LATIN SMALL LETTER D | ||||
|         '\ufffe',    //       0x85 -> REVERSED LATIN SMALL LETTER E | ||||
|         '\ufffe',    //       0x86 -> REVERSED LATIN SMALL LETTER F | ||||
|         '\ufffe',    //       0x87 -> REVERSED LATIN SMALL LETTER G | ||||
|         '\ufffe',    //       0x88 -> REVERSED LATIN SMALL LETTER H | ||||
|         '\ufffe',    //       0x89 -> REVERSED LATIN SMALL LETTER I | ||||
|         '\ufffe',    //       0x8A -> REVERSED LATIN SMALL LETTER J | ||||
|         '\ufffe',    //       0x8B -> REVERSED LATIN SMALL LETTER K | ||||
|         '\ufffe',    //       0x8C -> REVERSED LATIN SMALL LETTER L | ||||
|         '\ufffe',    //       0x8D -> REVERSED LATIN SMALL LETTER M | ||||
|         '\ufffe',    //       0x8E -> REVERSED LATIN SMALL LETTER N | ||||
|         '\ufffe',    //       0x8F -> REVERSED LATIN SMALL LETTER O | ||||
|         '\ufffe',    //       0x90 -> REVERSED LATIN SMALL LETTER P | ||||
|         '\ufffe',    //       0x91 -> REVERSED LATIN SMALL LETTER Q | ||||
|         '\ufffe',    //       0x92 -> REVERSED LATIN SMALL LETTER R | ||||
|         '\ufffe',    //       0x93 -> REVERSED LATIN SMALL LETTER S | ||||
|         '\ufffe',    //       0x94 -> REVERSED LATIN SMALL LETTER T | ||||
|         '\ufffe',    //       0x95 -> REVERSED LATIN SMALL LETTER U | ||||
|         '\ufffe',    //       0x96 -> REVERSED LATIN SMALL LETTER V | ||||
|         '\ufffe',    //       0x97 -> REVERSED LATIN SMALL LETTER W | ||||
|         '\ufffe',    //       0x98 -> REVERSED LATIN SMALL LETTER X | ||||
|         '\ufffe',    //       0x99 -> REVERSED LATIN SMALL LETTER Y | ||||
|         '\ufffe',    //       0x9A -> REVERSED LATIN SMALL LETTER Z | ||||
|         '\ufffe',    //       0x9B -> REVERSED LEFT SQUARE BRACKET | ||||
|         '\ufffe',    //       0x9C -> REVERSED REVERSE SOLIDUS | ||||
|         '\ufffe',    //       0x9D -> REVERSED RIGHT SQUARE BRACKET | ||||
|         '\ufffe',    //       0x9E -> REVERSED CIRCUMFLEX | ||||
|         '\ufffe',    //       0x9F -> REVERSED UNDERSCORE | ||||
|         '\ufffe',    //       0xA0 -> REVERSED SPACE | ||||
|         '\ufffe',    //       0xA1 -> REVERSED EXCLAMATION MARK | ||||
|         '\ufffe',    //       0xA2 -> REVERSED QUOTATION MARK | ||||
|         '\ufffe',    //       0xA3 -> REVERSED NUMBER SIGN | ||||
|         '\ufffe',    //       0xA4 -> REVERSED DOLLAR SIGN | ||||
|         '\ufffe',    //       0xA5 -> REVERSED PERCENT SIGN | ||||
|         '\ufffe',    //       0xA6 -> REVERSED AMPERSAND | ||||
|         '\ufffe',    //       0xA7 -> REVERSED APOSTROPHE | ||||
|         '\ufffe',    //       0xA8 -> REVERSED LEFT PARENTHESIS | ||||
|         '\ufffe',    //       0xA9 -> REVERSED RIGHT PARENTHESIS | ||||
|         '\ufffe',    //       0xAA -> REVERSED ASTERISK | ||||
|         '\ufffe',    //       0xAB -> REVERSED PLUS SIGN | ||||
|         '\ufffe',    //       0xAC -> REVERSED COMMA | ||||
|         '\ufffe',    //       0xAD -> REVERSED HYPHEN-MINUS | ||||
|         '\ufffe',    //       0xAE -> REVERSED FULL STOP | ||||
|         '\ufffe',    //       0xAF -> REVERSED SOLIDUS | ||||
|         '\ufffe',    //       0xB0 -> REVERSED DIGIT ZERO | ||||
|         '\ufffe',    //       0xB1 -> REVERSED DIGIT ONE | ||||
|         '\ufffe',    //       0xB2 -> REVERSED DIGIT TWO | ||||
|         '\ufffe',    //       0xB3 -> REVERSED DIGIT THREE | ||||
|         '\ufffe',    //       0xB4 -> REVERSED DIGIT FOUR | ||||
|         '\ufffe',    //       0xB5 -> REVERSED DIGIT FIVE | ||||
|         '\ufffe',    //       0xB6 -> REVERSED DIGIT SIX | ||||
|         '\ufffe',    //       0xB7 -> REVERSED DIGIT SEVEN | ||||
|         '\ufffe',    //       0xB8 -> REVERSED DIGIT EIGHT | ||||
|         '\ufffe',    //       0xB9 -> REVERSED DIGIT NINE | ||||
|         '\ufffe',    //       0xBA -> REVERSED COLON | ||||
|         '\ufffe',    //       0xBB -> REVERSED SEMICOLON | ||||
|         '\ufffe',    //       0xBC -> REVERSED LESS-THAN SIGN | ||||
|         '\ufffe',    //       0xBD -> REVERSED EQUALS SIGN | ||||
|         '\ufffe',    //       0xBE -> REVERSED GREATER-THAN SIGN | ||||
|         '\ufffe',    //       0xBF -> REVERSED QUESTION MARK | ||||
|         '\ufffe',    //       0xC0 -> REVERSED GRAVE ACCENT | ||||
|         '\ufffe',    //       0xC1 -> REVERSED LATIN CAPITAL LETTER A | ||||
|         '\ufffe',    //       0xC2 -> REVERSED LATIN CAPITAL LETTER B | ||||
|         '\ufffe',    //       0xC3 -> REVERSED LATIN CAPITAL LETTER C | ||||
|         '\ufffe',    //       0xC4 -> REVERSED LATIN CAPITAL LETTER D | ||||
|         '\ufffe',    //       0xC5 -> REVERSED LATIN CAPITAL LETTER E | ||||
|         '\ufffe',    //       0xC6 -> REVERSED LATIN CAPITAL LETTER F | ||||
|         '\ufffe',    //       0xC7 -> REVERSED LATIN CAPITAL LETTER G | ||||
|         '\ufffe',    //       0xC8 -> REVERSED LATIN CAPITAL LETTER H | ||||
|         '\ufffe',    //       0xC9 -> REVERSED LATIN CAPITAL LETTER I | ||||
|         '\ufffe',    //       0xCA -> REVERSED LATIN CAPITAL LETTER J | ||||
|         '\ufffe',    //       0xCB -> REVERSED LATIN CAPITAL LETTER K | ||||
|         '\ufffe',    //       0xCC -> REVERSED LATIN CAPITAL LETTER L | ||||
|         '\ufffe',    //       0xCD -> REVERSED LATIN CAPITAL LETTER M | ||||
|         '\ufffe',    //       0xCE -> REVERSED LATIN CAPITAL LETTER N | ||||
|         '\ufffe',    //       0xCF -> REVERSED LATIN CAPITAL LETTER O | ||||
|         '\ufffe',    //       0xD0 -> REVERSED LATIN CAPITAL LETTER P | ||||
|         '\ufffe',    //       0xD1 -> REVERSED LATIN CAPITAL LETTER Q | ||||
|         '\ufffe',    //       0xD2 -> REVERSED LATIN CAPITAL LETTER R | ||||
|         '\ufffe',    //       0xD3 -> REVERSED LATIN CAPITAL LETTER S | ||||
|         '\ufffe',    //       0xD4 -> REVERSED LATIN CAPITAL LETTER T | ||||
|         '\ufffe',    //       0xD5 -> REVERSED LATIN CAPITAL LETTER U | ||||
|         '\ufffe',    //       0xD6 -> REVERSED LATIN CAPITAL LETTER V | ||||
|         '\ufffe',    //       0xD7 -> REVERSED LATIN CAPITAL LETTER W | ||||
|         '\ufffe',    //       0xD8 -> REVERSED LATIN CAPITAL LETTER X | ||||
|         '\ufffe',    //       0xD9 -> REVERSED LATIN CAPITAL LETTER Y | ||||
|         '\ufffe',    //       0xDA -> REVERSED LATIN CAPITAL LETTER Z | ||||
|         '\ufffe',    //       0xDB -> REVERSED LEFT BRACE | ||||
|         '\ufffe',    //       0xDC -> REVERSED VERTICAL BAR | ||||
|         '\ufffe',    //       0xDD -> REVERSED RIGHT BRACE | ||||
|         '\ufffe',    //       0xDE -> REVERSED TILDE | ||||
|         '\ufffe',    //       0xDF -> RESERVED | ||||
|         '\ufffe',    //       0xE0 -> RESERVED | ||||
|         '\ufffe',    //       0xE1 -> REVERSED COMMODORE SYMBOL | ||||
|         '\ufffe',    //       0xE2 -> REVERSED UP ARROW | ||||
|         '\ufffe',    //       0xE3 -> REVERSED DOWN ARROW | ||||
|         '\ufffe',    //       0xE4 -> REVERSED LEFT ARROW | ||||
|         '\ufffe',    //       0xE5 -> REVERSED RIGHT ARROW | ||||
|         '\ufffe',    //       0xE6 -> REVERSED ANALOG CLOCKFACE | ||||
|         '\ufffe',    //       0xE7 -> REVERSED CYCLE ARROWS | ||||
|         '\ufffe',    //       0xE8 -> REVERSED ELLIPSIS | ||||
|         '\ufffe',    //       0xE9 -> REVERSED DIAGONAL STRIPES | ||||
|         '\ufffe',    //       0xEA -> REVERSED CHECKBOX UNCHECKED | ||||
|         '\ufffe',    //       0xEB -> REVERSED CHECKBOX CHECKED | ||||
|         '\ufffe',    //       0xEC -> REVERSED RADIO BUTTON UNSELECTED | ||||
|         '\ufffe',    //       0xED -> REVERSED RADIO BUTTON SELECTED | ||||
|         '\ufffe',    //       0xEE -> MEMORY CHIP ICON | ||||
|         '\u21e7',    //  ⇧    0xEF -> SHIFT SYMBOL | ||||
|         '\ufffe',    //       0xF0 -> REVERSED COPYRIGHT SYMBOL | ||||
|         '\ufffe',    //       0xF1 -> REVERSED CHECKMARK | ||||
|         '\ufffe',    //       0xF2 -> REVERSED THREE HORIZONTAL STRIPES | ||||
|         '\ufffe',    //       0xF3 -> REVERSED TICK TRACK | ||||
|         '\ufffe',    //       0xF4 -> REVERSED TICK TRACK NUB | ||||
|         '\ufffe',    //       0xF5 -> REVERSED TAB CORNER | ||||
|         '\ufffe',    //       0xF6 -> REVERSED THREE VERTICAL STRIPES | ||||
|         '\ufffe',    //       0xF7 -> CUSTOM 10 | ||||
|         '\ufffe',    //       0xF8 -> CUSTOM 11 | ||||
|         '\ufffe',    //       0xF9 -> CUSTOM 12 | ||||
|         '\ufffe',    //       0xFA -> CUSTOM 13 | ||||
|         '\ufffe',    //       0xFB -> CUSTOM 14 | ||||
|         '\ufffe',    //       0xFC -> CUSTOM 15 | ||||
|         '\ufffe',    //       0xFD -> CUSTOM 16 | ||||
|         '\ufffe',    //       0xFE -> CUSTOM 17 | ||||
|         '\ufffe'     //       0xFF -> CUSTOM 18 | ||||
|     ) | ||||
|  | ||||
|     // encoding:  from unicode to C64 OS Screencodes (0-255) | ||||
|     private val encodingC64os = decodingC64os.withIndex().associate{it.value to it.index} | ||||
|  | ||||
|     private fun replaceSpecial(chr: Char): Char = | ||||
|         when(chr) { | ||||
|             '\r' -> '\n'        // to make \r (carriage returrn) equivalent to \n (line feed): RETURN ($0d) | ||||
|             else -> chr | ||||
|         } | ||||
|      | ||||
|     fun encode(text: String, lowercase: Boolean = false): Result<List<UByte>, CharConversionException> { | ||||
|         fun encodeChar(chr3: Char, lowercase: Boolean): UByte { | ||||
|             val chr = replaceSpecial(chr3) | ||||
|             val screencode = encodingC64os[chr] | ||||
|             return screencode?.toUByte() ?: when (chr) { | ||||
|                 '\u0000' -> 0u | ||||
|                 '\n' -> 13u | ||||
|                 in '\u8000'..'\u80ff' -> { | ||||
|                     // special case: take the lower 8 bit hex value directly | ||||
|                     (chr.code - 0x8000).toUByte() | ||||
|                 } | ||||
|                 else -> { | ||||
|                     if(chr.isISOControl()) | ||||
|                         throw CharConversionException("no c64os character for char #${chr.code}") | ||||
|                     else | ||||
|                         throw CharConversionException("no c64os character for char #${chr.code} '${chr}'") | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         return try { | ||||
|             Ok(text.map { | ||||
|                 try { | ||||
|                     encodeChar(it, lowercase) | ||||
|                 } catch (x: CharConversionException) { | ||||
|                     encodeChar(it, !lowercase) | ||||
|                 } | ||||
|             }) | ||||
|         } catch(cx: CharConversionException) { | ||||
|             Err(cx) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun decode(screencode: Iterable<UByte>, lowercase: Boolean = false): Result<String, CharConversionException> { | ||||
|         return try { | ||||
|             Ok(screencode.map { | ||||
|                 val code = it.toInt() | ||||
|                 if(code<0 || code>= decodingC64os.size) | ||||
|                     throw CharConversionException("c64os $code out of range 0..${decodingC64os.size-1}") | ||||
|                 decodingC64os[code] | ||||
|             }.joinToString("")) | ||||
|         } catch(ce: CharConversionException) { | ||||
|             Err(ce) | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										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) | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										47
									
								
								codeCore/src/prog8/code/target/encodings/Encoder.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										47
									
								
								codeCore/src/prog8/code/target/encodings/Encoder.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,47 @@ | ||||
| package prog8.code.target.encodings | ||||
|  | ||||
| import com.github.michaelbull.result.fold | ||||
| import prog8.code.core.Encoding | ||||
| import prog8.code.core.IStringEncoding | ||||
| import prog8.code.core.InternalCompilerException | ||||
|  | ||||
| class Encoder(val newlineToCarriageReturn: Boolean): IStringEncoding { | ||||
|     override val defaultEncoding: Encoding = Encoding.ISO | ||||
|  | ||||
|     override fun encodeString(str: String, encoding: Encoding): List<UByte> { | ||||
|         val coded = when(encoding) { | ||||
|             Encoding.PETSCII -> PetsciiEncoding.encodePetscii(str, true) | ||||
|             Encoding.SCREENCODES -> PetsciiEncoding.encodeScreencode(str, true) | ||||
|             Encoding.ISO -> IsoEncoding.encode(str, newlineToCarriageReturn) | ||||
|             Encoding.ISO5 -> IsoCyrillicEncoding.encode(str, newlineToCarriageReturn) | ||||
|             Encoding.ISO16 -> IsoEasternEncoding.encode(str, newlineToCarriageReturn) | ||||
|             Encoding.CP437 -> Cp437Encoding.encode(str) | ||||
|             Encoding.KATAKANA -> KatakanaEncoding.encode(str, newlineToCarriageReturn) | ||||
|             Encoding.ATASCII -> AtasciiEncoding.encode(str) | ||||
|             Encoding.C64OS -> C64osEncoding.encode(str) | ||||
|             else -> throw InternalCompilerException("unsupported encoding $encoding") | ||||
|         } | ||||
|         return coded.fold( | ||||
|             failure = { throw it }, | ||||
|             success = { it } | ||||
|         ) | ||||
|     } | ||||
|     override fun decodeString(bytes: Iterable<UByte>, encoding: Encoding): String { | ||||
|         val decoded = when(encoding) { | ||||
|             Encoding.PETSCII -> PetsciiEncoding.decodePetscii(bytes, true) | ||||
|             Encoding.SCREENCODES -> PetsciiEncoding.decodeScreencode(bytes, true) | ||||
|             Encoding.ISO -> IsoEncoding.decode(bytes, newlineToCarriageReturn) | ||||
|             Encoding.ISO5 -> IsoCyrillicEncoding.decode(bytes, newlineToCarriageReturn) | ||||
|             Encoding.ISO16 -> IsoEasternEncoding.decode(bytes, newlineToCarriageReturn) | ||||
|             Encoding.CP437 -> Cp437Encoding.decode(bytes) | ||||
|             Encoding.KATAKANA -> KatakanaEncoding.decode(bytes, newlineToCarriageReturn) | ||||
|             Encoding.ATASCII -> AtasciiEncoding.decode(bytes) | ||||
|             Encoding.C64OS -> C64osEncoding.decode(bytes) | ||||
|             else -> throw InternalCompilerException("unsupported encoding $encoding") | ||||
|         } | ||||
|         return decoded.fold( | ||||
|             failure = { throw it }, | ||||
|             success = { it } | ||||
|         ) | ||||
|     } | ||||
| } | ||||
							
								
								
									
										49
									
								
								codeCore/src/prog8/code/target/encodings/IsoEncoding.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								codeCore/src/prog8/code/target/encodings/IsoEncoding.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| 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 | ||||
|  | ||||
|  | ||||
| open class IsoEncodingBase(charsetName: String) { | ||||
|     val charset: Charset = Charset.forName(charsetName) | ||||
|  | ||||
|     fun encode(str: String, newlineToCarriageReturn: Boolean): Result<List<UByte>, CharConversionException> { | ||||
|         return try { | ||||
|             val mapped = str.map { chr -> | ||||
|                 when (chr) { | ||||
|                     '\u0000' -> 0u | ||||
|                     '\n' -> if(newlineToCarriageReturn) 13u else 10u | ||||
|                     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>, newlineToCarriageReturn: Boolean): Result<String, CharConversionException> { | ||||
|         return try { | ||||
|             Ok(String(bytes.map { | ||||
|                 when(it) { | ||||
|                     13u.toUByte() -> if(newlineToCarriageReturn) 10 else 13 | ||||
|                     else -> it.toByte() | ||||
|                 } | ||||
|             }.toByteArray(), charset)) | ||||
|         } catch (ce: CharConversionException) { | ||||
|             Err(ce) | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
| object IsoEncoding: IsoEncodingBase("ISO-8859-15") | ||||
| object IsoCyrillicEncoding: IsoEncodingBase("ISO-8859-5") | ||||
| object IsoEasternEncoding: IsoEncodingBase("ISO-8859-16") | ||||
							
								
								
									
										128
									
								
								codeCore/src/prog8/code/target/encodings/KatakanaEncoding.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								codeCore/src/prog8/code/target/encodings/KatakanaEncoding.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | ||||
| 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, newlineToCarriageReturn: Boolean): Result<List<UByte>, CharConversionException> { | ||||
|         return try { | ||||
|             val mapped = str.map { chr -> | ||||
|                 when (chr) { | ||||
|                     '\n' -> if(newlineToCarriageReturn) 13u else 10u | ||||
|  | ||||
|                     '\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>, newlineToCarriageReturn: Boolean): Result<String, CharConversionException> { | ||||
|         return try { | ||||
|             Ok(String(bytes.map { | ||||
|                 when(it) { | ||||
|                     13u.toUByte() -> if(newlineToCarriageReturn) 10 else 13 | ||||
|                     else -> it.toByte() | ||||
|                 } | ||||
|             }.toByteArray(), charset)) | ||||
|         } catch (ce: CharConversionException) { | ||||
|             Err(ce) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -1,15 +1,14 @@ | ||||
| package prog8.compiler.target.cbm | ||||
| package prog8.code.target.encodings | ||||
| 
 | ||||
| import com.github.michaelbull.result.Err | ||||
| import com.github.michaelbull.result.Ok | ||||
| import com.github.michaelbull.result.Result | ||||
| import prog8.ast.antlr.escape | ||||
| import java.io.CharConversionException | ||||
| 
 | ||||
| object Petscii { | ||||
| object PetsciiEncoding { | ||||
| 
 | ||||
|     // decoding:  from Petscii/Screencodes (0-255) to unicode | ||||
|     // character tables used from https://github.com/dj51d/cbmcodecs | ||||
|     // decoding:  from Petscii/Screencodes (0-255) to Unicode | ||||
|     // character tables used from https://github.com/irmen/cbmcodecs2 | ||||
| 
 | ||||
|     private val decodingPetsciiLowercase = charArrayOf( | ||||
|         '\u0000',    //       0x00 -> \u0000 | ||||
| @@ -22,10 +21,10 @@ object Petscii { | ||||
|         '\ufffe',    //       0x07 -> UNDEFINED | ||||
|         '\uf118',    //       0x08 -> DISABLE CHARACTER SET SWITCHING (CUS) | ||||
|         '\uf119',    //       0x09 -> ENABLE CHARACTER SET SWITCHING (CUS) | ||||
|         '\ufffe',    //       0x0A -> UNDEFINED | ||||
|         '\n',        //       0x0A -> LINE FEED (RETURN) | ||||
|         '\ufffe',    //       0x0B -> UNDEFINED | ||||
|         '\ufffe',    //       0x0C -> UNDEFINED | ||||
|         '\r'    ,    //       0x0D -> CARRIAGE RETURN | ||||
|         '\n'    ,    //       0x0D -> LINE FEED (RETURN) | ||||
|         '\u000e',    //       0x0E -> SHIFT OUT | ||||
|         '\ufffe',    //       0x0F -> UNDEFINED | ||||
|         '\ufffe',    //       0x10 -> UNDEFINED | ||||
| @@ -153,13 +152,13 @@ object Petscii { | ||||
|         '\uf113',    //      0x8A -> FUNCTION KEY 4 (CUS) | ||||
|         '\uf115',    //      0x8B -> FUNCTION KEY 6 (CUS) | ||||
|         '\uf117',    //      0x8C -> FUNCTION KEY 8 (CUS) | ||||
|         '\n'    ,    //       0x8D -> LINE FEED | ||||
|         '\r'    ,    //       0x8D -> CARRIAGE RETURN (SHIFT-RETURN) | ||||
|         '\u000f',    //      0x8E -> SHIFT IN | ||||
|         '\ufffe',    //       0x8F -> UNDEFINED | ||||
|         '\uf105',    //       0x90 -> BLACK COLOR SWITCH (CUS) | ||||
|         '\uf11e',    //      0x91 -> CURSOR UP (CUS) | ||||
|         '\uf11b',    //      0x92 -> REVERSE VIDEO OFF (CUS) | ||||
|         '\u000c',    //       0x93 -> FORM FEED | ||||
|         '\u000c',    //       0x93 -> FORM FEED (CLEAR SCREEN) | ||||
|         '\uf121',    //      0x94 -> INSERT (CUS) | ||||
|         '\uf106',    //       0x95 -> BROWN COLOR SWITCH (CUS) | ||||
|         '\uf107',    //       0x96 -> LIGHT RED COLOR SWITCH (CUS) | ||||
| @@ -195,7 +194,7 @@ object Petscii { | ||||
|         '\u258e',    //  ▎    0xB4 -> LEFT ONE QUARTER BLOCK | ||||
|         '\u258d',    //  ▍    0xB5 -> LEFT THREE EIGTHS BLOCK | ||||
|         '\uf131',    //      0xB6 -> RIGHT THREE EIGHTHS BLOCK (CUS) | ||||
|         '\uf132',    //      0xB7 -> UPPER ONE QUARTER BLOCK (CUS) | ||||
|         '\uf132',    //  🮂    0xB7 -> UPPER ONE QUARTER BLOCK (CUS) | ||||
|         '\uf133',    //      0xB8 -> UPPER THREE EIGHTS BLOCK (CUS) | ||||
|         '\u2583',    //  ▃    0xB9 -> LOWER THREE EIGHTHS BLOCK | ||||
|         '\u2713',    //  ✓    0xBA -> CHECK MARK | ||||
| @@ -246,7 +245,7 @@ object Petscii { | ||||
|         '\u2595',    //  ▕    0xE7 -> RIGHT ONE EIGHTH BLOCK | ||||
|         '\uf12f',    //      0xE8 -> LOWER HALF BLOCK MEDIUM SHADE (CUS) | ||||
|         '\uf13a',    //      0xE9 -> MEDIUM SHADE SLASHED RIGHT (CUS) | ||||
|         '\uf130',    //      0xEA -> RIGHT ONE QUARTER BLOCK (CUS) | ||||
|         '\uf130',    //  🮇    0xEA -> RIGHT ONE QUARTER BLOCK (CUS) | ||||
|         '\u251c',    //  ├    0xEB -> BOX DRAWINGS LIGHT VERTICAL AND RIGHT | ||||
|         '\u2597',    //  ▗    0xEC -> QUADRANT LOWER RIGHT | ||||
|         '\u2514',    //  └    0xED -> BOX DRAWINGS LIGHT UP AND RIGHT | ||||
| @@ -284,7 +283,7 @@ object Petscii { | ||||
|         '\ufffe',    //       0x0A -> UNDEFINED | ||||
|         '\ufffe',    //       0x0B -> UNDEFINED | ||||
|         '\ufffe',    //       0x0C -> UNDEFINED | ||||
|         '\r'    ,    //       0x0D -> CARRIAGE RETURN | ||||
|         '\n'    ,    //       0x0D -> LINE FEED (RETURN) | ||||
|         '\u000e',    //       0x0E -> SHIFT OUT | ||||
|         '\ufffe',    //       0x0F -> UNDEFINED | ||||
|         '\ufffe',    //       0x10 -> UNDEFINED | ||||
| @@ -412,13 +411,13 @@ object Petscii { | ||||
|         '\uf113',    //       0x8A -> FUNCTION KEY 4 (CUS) | ||||
|         '\uf115',    //       0x8B -> FUNCTION KEY 6 (CUS) | ||||
|         '\uf117',    //       0x8C -> FUNCTION KEY 8 (CUS) | ||||
|         '\n'    ,    //       0x8D -> LINE FEED | ||||
|         '\r'    ,    //       0x8D -> CARRIAGE RETURN (SHIFT-RETURN) | ||||
|         '\u000f',    //       0x8E -> SHIFT IN | ||||
|         '\ufffe',    //       0x8F -> UNDEFINED | ||||
|         '\uf105',    //       0x90 -> BLACK COLOR SWITCH (CUS) | ||||
|         '\uf11e',    //       0x91 -> CURSOR UP (CUS) | ||||
|         '\uf11b',    //       0x92 -> REVERSE VIDEO OFF (CUS) | ||||
|         '\u000c',    //       0x93 -> FORM FEED | ||||
|         '\u000c',    //       0x93 -> FORM FEED (CLEAR SCREEN) | ||||
|         '\uf121',    //       0x94 -> INSERT (CUS) | ||||
|         '\uf106',    //       0x95 -> BROWN COLOR SWITCH (CUS) | ||||
|         '\uf107',    //       0x96 -> LIGHT RED COLOR SWITCH (CUS) | ||||
| @@ -1062,6 +1061,7 @@ object Petscii { | ||||
|             '}' -> '├' | ||||
|             '|' -> '│' | ||||
|             '\\' -> '╲' | ||||
|             '\r' -> '\n'        // to make \r (carriage returrn) equivalent to \n (line feed): RETURN ($0d) | ||||
|             else -> chr | ||||
|         } | ||||
| 
 | ||||
| @@ -1077,7 +1077,10 @@ object Petscii { | ||||
|                 } | ||||
|                 else -> { | ||||
|                     val case = if (lowercase) "lower" else "upper" | ||||
|                     throw CharConversionException("no ${case}Petscii character for '${escape(chr.toString())}' (${chr.code})") | ||||
|                     if(chr.isISOControl()) | ||||
|                         throw CharConversionException("no ${case}Petscii character for char #${chr.code}") | ||||
|                     else | ||||
|                         throw CharConversionException("no ${case}Petscii character for char #${chr.code} '${chr}'") | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| @@ -1086,7 +1089,7 @@ object Petscii { | ||||
|             Ok(text.map { | ||||
|                 try { | ||||
|                     encodeChar(it, lowercase) | ||||
|                 } catch (x: CharConversionException) { | ||||
|                 } catch (_: CharConversionException) { | ||||
|                     encodeChar(it, !lowercase) | ||||
|                 } | ||||
|             }) | ||||
| @@ -1095,13 +1098,17 @@ object Petscii { | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun decodePetscii(petscii: Iterable<UByte>, lowercase: Boolean = false): String { | ||||
|         return petscii.map { | ||||
|             val code = it.toInt() | ||||
|             if(code<0 || code>= decodingPetsciiLowercase.size) | ||||
|                 throw CharConversionException("petscii $code out of range 0..${decodingPetsciiLowercase.size-1}") | ||||
|             if(lowercase) decodingPetsciiLowercase[code] else decodingPetsciiUppercase[code] | ||||
|         }.joinToString("") | ||||
|     fun decodePetscii(petscii: Iterable<UByte>, lowercase: Boolean = false): Result<String, CharConversionException> { | ||||
|         return try { | ||||
|             Ok(petscii.map { | ||||
|                 val code = it.toInt() | ||||
|                 if(code<0 || code>= decodingPetsciiLowercase.size) | ||||
|                     throw CharConversionException("petscii $code out of range 0..${decodingPetsciiLowercase.size-1}") | ||||
|                 if(lowercase) decodingPetsciiLowercase[code] else decodingPetsciiUppercase[code] | ||||
|             }.joinToString("")) | ||||
|         } catch(ce: CharConversionException) { | ||||
|             return Err(ce) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun encodeScreencode(text: String, lowercase: Boolean = false): Result<List<UByte>, CharConversionException> { | ||||
| @@ -1110,13 +1117,18 @@ object Petscii { | ||||
|             val screencode = if(lowercase) encodingScreencodeLowercase[chr] else encodingScreencodeUppercase[chr] | ||||
|             return screencode?.toUByte() ?: when (chr) { | ||||
|                 '\u0000' -> 0u | ||||
|                 '\n' -> 141u | ||||
|                 '\r' -> 141u | ||||
|                 in '\u8000'..'\u80ff' -> { | ||||
|                     // special case: take the lower 8 bit hex value directly | ||||
|                     (chr.code - 0x8000).toUByte() | ||||
|                 } | ||||
|                 else -> { | ||||
|                     val case = if (lowercase) "lower" else "upper" | ||||
|                     throw CharConversionException("no ${case}Screencode character for '${escape(chr.toString())}' (${chr.code})") | ||||
|                     if(chr.isISOControl()) | ||||
|                         throw CharConversionException("no ${case}Screencode character for char #${chr.code}") | ||||
|                     else | ||||
|                         throw CharConversionException("no ${case}Screencode character for char #${chr.code} '${chr}'") | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| @@ -1125,7 +1137,7 @@ object Petscii { | ||||
|             Ok(text.map { | ||||
|                 try { | ||||
|                     encodeChar(it, lowercase) | ||||
|                 } catch (x: CharConversionException) { | ||||
|                 } catch (_: CharConversionException) { | ||||
|                     encodeChar(it, !lowercase) | ||||
|                 } | ||||
|             }) | ||||
| @@ -1134,25 +1146,29 @@ object Petscii { | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun decodeScreencode(screencode: Iterable<UByte>, lowercase: Boolean = false): String { | ||||
|         return screencode.map { | ||||
|             val code = it.toInt() | ||||
|             if(code<0 || code>= decodingScreencodeLowercase.size) | ||||
|                 throw CharConversionException("screencode $code out of range 0..${decodingScreencodeLowercase.size-1}") | ||||
|             if (lowercase) decodingScreencodeLowercase[code] else decodingScreencodeUppercase[code] | ||||
|         }.joinToString("") | ||||
|     fun decodeScreencode(screencode: Iterable<UByte>, lowercase: Boolean = false): Result<String, CharConversionException> { | ||||
|         return try { | ||||
|             Ok(screencode.map { | ||||
|                 val code = it.toInt() | ||||
|                 if(code<0 || code>= decodingScreencodeLowercase.size) | ||||
|                     throw CharConversionException("screencode $code out of range 0..${decodingScreencodeLowercase.size-1}") | ||||
|                 if (lowercase) decodingScreencodeLowercase[code] else decodingScreencodeUppercase[code] | ||||
|             }.joinToString("")) | ||||
|         } catch(ce: CharConversionException) { | ||||
|             Err(ce) | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fun petscii2scr(petscii_code: UByte, inverseVideo: Boolean): Result<UByte, CharConversionException> { | ||||
|     fun petscii2scr(petsciicode: UByte, inverseVideo: Boolean): Result<UByte, CharConversionException> { | ||||
|         val code: UInt = when { | ||||
|             petscii_code <= 0x1fu -> petscii_code + 128u | ||||
|             petscii_code <= 0x3fu -> petscii_code.toUInt() | ||||
|             petscii_code <= 0x5fu -> petscii_code - 64u | ||||
|             petscii_code <= 0x7fu -> petscii_code - 32u | ||||
|             petscii_code <= 0x9fu -> petscii_code + 64u | ||||
|             petscii_code <= 0xbfu -> petscii_code - 64u | ||||
|             petscii_code <= 0xfeu -> petscii_code - 128u | ||||
|             petscii_code == 255.toUByte() -> 95u | ||||
|             petsciicode <= 0x1fu -> petsciicode + 128u | ||||
|             petsciicode <= 0x3fu -> petsciicode.toUInt() | ||||
|             petsciicode <= 0x5fu -> petsciicode - 64u | ||||
|             petsciicode <= 0x7fu -> petsciicode - 32u | ||||
|             petsciicode <= 0x9fu -> petsciicode + 64u | ||||
|             petsciicode <= 0xbfu -> petsciicode - 64u | ||||
|             petsciicode <= 0xfeu -> petsciicode - 128u | ||||
|             petsciicode == 255.toUByte() -> 95u | ||||
|             else -> return Err(CharConversionException("petscii code out of range")) | ||||
|         } | ||||
|         if(inverseVideo) { | ||||
							
								
								
									
										69
									
								
								codeCore/src/prog8/code/target/zp/C128Zeropage.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								codeCore/src/prog8/code/target/zp/C128Zeropage.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| package prog8.code.target.zp | ||||
|  | ||||
| import prog8.code.core.CompilationOptions | ||||
| import prog8.code.core.InternalCompilerException | ||||
| import prog8.code.core.Zeropage | ||||
| import prog8.code.core.ZeropageType | ||||
|  | ||||
|  | ||||
| // reference: "Mapping the C128" zeropage chapter. | ||||
|  | ||||
| class C128Zeropage(options: CompilationOptions) : Zeropage(options) { | ||||
|  | ||||
|     override val SCRATCH_B1 = 0x74u      // temp storage for a single byte | ||||
|     override val SCRATCH_REG = 0x75u     // temp storage for a register byte, must be B1+1 | ||||
|     override val SCRATCH_W1 = 0xfbu      // temp storage 1 for a word  $fb+$fc | ||||
|     override val SCRATCH_W2 = 0xfdu      // temp storage 2 for a word  $fd+$fe | ||||
|     override val SCRATCH_PTR = 0x0bu     // temp storage for a pointer $0b+$0c | ||||
|  | ||||
|     init { | ||||
|         if (options.floats && options.zeropage !in arrayOf( | ||||
|                 ZeropageType.FLOATSAFE, | ||||
|                 ZeropageType.BASICSAFE, | ||||
|                 ZeropageType.DONTUSE | ||||
|             )) | ||||
|             throw InternalCompilerException("when floats are enabled, zero page type should be 'floatsafe' or 'basicsafe' or 'dontuse'") | ||||
|  | ||||
|         when (options.zeropage) { | ||||
|             ZeropageType.FULL -> { | ||||
|                 // $00/$01 are data port IO registers, // $02-$09 are storage locations for JSRFAR and such | ||||
|                 free.addAll(0x0au..0xffu) | ||||
|                 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 -> { | ||||
|                 free.addAll(0x0au..0x8fu)       // BASIC variables | ||||
|                 free.addAll(arrayOf(0x92u, 0x96u, 0x9bu, 0x9cu, 0x9eu, 0x9fu, 0xa4u, 0xa7u, 0xa8u, 0xa9u, 0xaau, 0xabu, | ||||
|                     0xb0u, 0xb1u, 0xb4u, 0xb5u, 0xb6u)) | ||||
|             } | ||||
|             ZeropageType.FLOATSAFE, | ||||
|             ZeropageType.BASICSAFE -> { | ||||
|                 free.addAll(arrayOf(0x0bu, 0x0cu, 0x0du, 0x0eu, 0x0fu, 0x10u, 0x11u, 0x12u, 0x16u, 0x17u, 0x18u, 0x19u, 0x1au)) | ||||
|                 free.addAll(0x1bu..0x23u) | ||||
|                 free.addAll(arrayOf(0x3fu, 0x40u, 0x41u, 0x42u, 0x43u, 0x44u, 0x47u, 0x48u, 0x49u, 0x4au, 0x4bu, 0x4cu, 0x4fu, | ||||
|                     0x55u, 0x56u, 0x57u, 0x58u, | ||||
|                     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, | ||||
|                     0xb0u, 0xb1u, 0xb4u, 0xb5u, 0xb6u | ||||
|                 )) | ||||
|  | ||||
|                 // 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 | ||||
|                     free.addAll(arrayOf(0x14u, 0x28u, 0x29u, 0x2au, 0x2bu, 0x2cu, | ||||
|                         0x50u, 0x51u, 0x52u, 0x53u, 0x54u, 0x59u, 0x5au, 0x5bu, 0x5cu, 0x5du, 0x5eu, 0x5fu, 0x60u, 0x61u, 0x62u, | ||||
|                         0x63u, 0x64u, 0x65u, 0x66u, 0x67u, 0x68u, | ||||
|                         0x6au, 0x6bu, 0x6cu, 0x6du, 0x6eu, 0x6fu, 0x71u)) | ||||
|                 // } | ||||
|             } | ||||
|             ZeropageType.DONTUSE -> { | ||||
|                 free.clear()  // don't use zeropage at all | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val distinctFree = free.distinct() | ||||
|         free.clear() | ||||
|         free.addAll(distinctFree) | ||||
|  | ||||
|         removeReservedFromFreePool() | ||||
|         retainAllowed() | ||||
|     } | ||||
| } | ||||
							
								
								
									
										98
									
								
								codeCore/src/prog8/code/target/zp/C64Zeropage.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								codeCore/src/prog8/code/target/zp/C64Zeropage.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,98 @@ | ||||
| package prog8.code.target.zp | ||||
|  | ||||
| import prog8.code.core.* | ||||
|  | ||||
|  | ||||
| class C64Zeropage(options: CompilationOptions) : Zeropage(options) { | ||||
|  | ||||
|     override val SCRATCH_B1 = 0x02u      // temp storage for a single byte | ||||
|     override val SCRATCH_REG = 0x03u     // temp storage for a register byte, must be B1+1 | ||||
|     override val SCRATCH_W1 = 0xfbu      // temp storage 1 for a word  $fb+$fc | ||||
|     override val SCRATCH_W2 = 0xfdu      // temp storage 2 for a word  $fd+$fe | ||||
|     override val SCRATCH_PTR = 0x9bu     // temp storage for a pointer $9b+$9c | ||||
|  | ||||
|  | ||||
|     init { | ||||
|         if (options.floats && options.zeropage !in arrayOf( | ||||
|                 ZeropageType.FLOATSAFE, | ||||
|                 ZeropageType.BASICSAFE, | ||||
|                 ZeropageType.DONTUSE | ||||
|             )) | ||||
|             throw InternalCompilerException("when floats are enabled, zero page type should be 'floatsafe' or 'basicsafe' or 'dontuse'") | ||||
|  | ||||
|         if (options.zeropage == ZeropageType.FULL) { | ||||
|             free.addAll(0x02u..0xffu) | ||||
|             free.removeAll(arrayOf(0xa0u, 0xa1u, 0xa2u, 0x91u, 0xc0u, 0xc5u, 0xcbu, 0xf5u, 0xf6u))        // these are updated by IRQ | ||||
|         } else { | ||||
|             if (options.zeropage == ZeropageType.KERNALSAFE || options.zeropage == ZeropageType.FLOATSAFE) { | ||||
|                 free.addAll(arrayOf( | ||||
|                         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, | ||||
|                         0x20, 0x21, 0x22, 0x23, 0x24, 0x25, | ||||
|                         0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e, 0x3f, 0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46, | ||||
|                         0x47, 0x48, 0x4b, 0x4c, 0x4d, 0x4e, 0x4f, 0x51, 0x52, 0x53, | ||||
|                         0x57, 0x58, 0x59, 0x5a, 0x5b, 0x5c, 0x5d, 0x5e, 0x5f, 0x60, | ||||
|                         0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, | ||||
|                         0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f, 0x70, 0x71, 0x72, | ||||
|                         0x73, 0x74, 0x75, 0x76, 0x77, 0x78, 0x79, 0x7a, 0x7b, 0x7c, | ||||
|                         0x7d, 0x7e, 0x7f, 0x80, 0x81, 0x82, 0x83, 0x84, 0x85, 0x86, 0x87, 0x88, 0x89, 0x8a, | ||||
|                         0x8b, 0x8c, 0x8d, 0x8e, 0x8f, 0xff | ||||
|                         // 0x90-0xfa is 'kernal work storage area' | ||||
|                 ).map{it.toUInt()}) | ||||
|             } | ||||
|  | ||||
|             if (options.zeropage == ZeropageType.FLOATSAFE) { | ||||
|                 // remove the zeropage locations used for floating point operations from the free list | ||||
|                 free.removeAll(arrayOf( | ||||
|                         0x03, 0x04, 0x05, 0x06, 0x10, 0x11, 0x12, | ||||
|                         0x22, 0x23, 0x24, 0x25, 0x26, 0x27, 0x28, 0x29, 0x2a, | ||||
|                         0x57, 0x58, 0x59, 0x5a, 0x5b, 0x5c, 0x5d, 0x5e, 0x5f, 0x60, | ||||
|                         0x61, 0x62, 0x63, 0x64, 0x65, 0x66, 0x67, 0x68, | ||||
|                         0x69, 0x6a, 0x6b, 0x6c, 0x6d, 0x6e, 0x6f, 0x70, 0x71, 0x72, | ||||
|                         0x8b, 0x8c, 0x8d, 0x8e, 0x8f, 0xff | ||||
|                 ).map{it.toUInt()}) | ||||
|             } | ||||
|  | ||||
|             if(options.zeropage != ZeropageType.DONTUSE) { | ||||
|                 // 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* | ||||
|                 free.addAll(arrayOf(0x02, 0x03, 0x04, 0x05, 0x06, 0x0a, 0x0e, | ||||
|                         0x92, 0x96, 0x9b, 0x9c, 0x9e, 0x9f, 0xa6, | ||||
|                         0xb0, 0xb1, 0xbe, 0xbf, 0xf9).map{it.toUInt()}) | ||||
|             } else { | ||||
|                 // don't use the zeropage at all | ||||
|                 free.clear() | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val distinctFree = free.distinct() | ||||
|         free.clear() | ||||
|         free.addAll(distinctFree) | ||||
|  | ||||
|         removeReservedFromFreePool() | ||||
|  | ||||
|         if(options.zeropage==ZeropageType.FULL || options.zeropage==ZeropageType.KERNALSAFE) { | ||||
|             // in these cases there is enough space on the zero page to stick the cx16 virtual registers in there as well. | ||||
|             allocateCx16VirtualRegisters() | ||||
|         } | ||||
|  | ||||
|         retainAllowed() | ||||
|     } | ||||
|  | ||||
|     private 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) | ||||
|         // The base addres is $04.  Unfortunately it cannot be the same as on the Commander X16 ($02). | ||||
|         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}s"]  = VarAllocation((4+reg*2).toUInt(), DataType.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}H"]  = VarAllocation((5+reg*2).toUInt(), DataType.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}sH"] = VarAllocation((5+reg*2).toUInt(), DataType.BYTE, 1)        // cx16.r0sH .. cx16.r15sH | ||||
|             free.remove((4+reg*2).toUInt()) | ||||
|             free.remove((5+reg*2).toUInt()) | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										69
									
								
								codeCore/src/prog8/code/target/zp/CX16Zeropage.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								codeCore/src/prog8/code/target/zp/CX16Zeropage.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| package prog8.code.target.zp | ||||
|  | ||||
| import prog8.code.core.* | ||||
|  | ||||
|  | ||||
| class CX16Zeropage(options: CompilationOptions) : Zeropage(options) { | ||||
|  | ||||
|     override val SCRATCH_B1 = 0x7au      // temp storage for a single byte | ||||
|     override val SCRATCH_REG = 0x7bu     // temp storage for a register byte, must be B1+1 | ||||
|     override val SCRATCH_W1 = 0x7cu      // temp storage 1 for a word  $7c+$7d | ||||
|     override val SCRATCH_W2 = 0x7eu      // temp storage 2 for a word  $7e+$7f | ||||
|     override val SCRATCH_PTR = 0x22u     // temp storage for a pointer $22+$23 | ||||
|  | ||||
|  | ||||
|     init { | ||||
|         if (options.floats && options.zeropage !in arrayOf( | ||||
|                 ZeropageType.FLOATSAFE, | ||||
|                 ZeropageType.BASICSAFE, | ||||
|                 ZeropageType.DONTUSE | ||||
|             )) | ||||
|             throw InternalCompilerException("when floats are enabled, zero page type should be 'floatsafe' or 'basicsafe' or 'dontuse'") | ||||
|  | ||||
|         // the addresses 0x02 to 0x21 (inclusive) are taken for sixteen virtual 16-bit api registers. | ||||
|  | ||||
|         synchronized(this) { | ||||
|             when (options.zeropage) { | ||||
|                 ZeropageType.FULL -> { | ||||
|                     free.addAll(0x22u..0xffu) | ||||
|                 } | ||||
|                 ZeropageType.KERNALSAFE -> { | ||||
|                     free.addAll(0x22u..0x7fu) | ||||
|                     free.addAll(0xa9u..0xffu) | ||||
|                 } | ||||
|                 ZeropageType.FLOATSAFE -> { | ||||
|                     free.addAll(0x22u..0x7fu) | ||||
|                     free.addAll(0xd4u..0xffu) | ||||
|                 } | ||||
|                 ZeropageType.BASICSAFE -> { | ||||
|                     free.addAll(0x22u..0x7fu) | ||||
|                 } | ||||
|                 ZeropageType.DONTUSE -> { | ||||
|                     free.clear() // don't use zeropage at all | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             val distinctFree = free.distinct() | ||||
|             free.clear() | ||||
|             free.addAll(distinctFree) | ||||
|  | ||||
|             removeReservedFromFreePool() | ||||
|             allocateCx16VirtualRegisters() | ||||
|             retainAllowed() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private 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.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}L"]  = VarAllocation((2+reg*2).toUInt(), DataType.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}sL"] = VarAllocation((2+reg*2).toUInt(), DataType.BYTE, 1)        // cx16.r0sL .. cx16.r15sL | ||||
|             allocatedVariables["cx16.r${reg}sH"] = VarAllocation((3+reg*2).toUInt(), DataType.BYTE, 1)        // cx16.r0sH .. cx16.r15sH | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										63
									
								
								codeCore/src/prog8/code/target/zp/ConfigurableZeropage.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								codeCore/src/prog8/code/target/zp/ConfigurableZeropage.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| package prog8.code.target.zp | ||||
|  | ||||
| import prog8.code.core.CompilationOptions | ||||
| import prog8.code.core.DataType | ||||
| import prog8.code.core.Zeropage | ||||
| import prog8.code.core.ZeropageType | ||||
|  | ||||
| class ConfigurableZeropage( | ||||
|     override val SCRATCH_B1: UInt,      // temp storage for a single byte | ||||
|     override val SCRATCH_REG: UInt,     // temp storage for a register byte, must be B1+1 | ||||
|     override val SCRATCH_W1: UInt,      // temp storage 1 for a word | ||||
|     override val SCRATCH_W2: UInt,      // temp storage 2 for a word | ||||
|     override val SCRATCH_PTR: UInt,     // temp storage for a pointer | ||||
|  | ||||
|     val virtualRegistersStart: UInt,        // location of 32 bytes for the r0-r15 virtual registers | ||||
|     basicsafe: List<UIntRange>, | ||||
|     kernalsafe: List<UIntRange>, | ||||
|     fullsafe: List<UIntRange>, | ||||
|     options: CompilationOptions | ||||
| ) : Zeropage(options) { | ||||
|  | ||||
|     init { | ||||
|         if (options.floats) { | ||||
|             TODO("floats in configurable target zp") | ||||
|         } | ||||
|  | ||||
|         if(SCRATCH_REG!=SCRATCH_B1+1u) | ||||
|             throw IllegalArgumentException("Zero page scratch variable REG should be B1+1") | ||||
|  | ||||
|         when (options.zeropage) { | ||||
|             ZeropageType.DONTUSE -> { /* don't use any zeropage at all */ } | ||||
|             ZeropageType.FULL -> fullsafe.forEach { free.addAll(it) } | ||||
|             ZeropageType.BASICSAFE -> basicsafe.forEach { free.addAll(it) } | ||||
|             ZeropageType.KERNALSAFE -> kernalsafe.forEach { free.addAll(it) } | ||||
|             ZeropageType.FLOATSAFE -> TODO("floatsafe in configurable target zp") | ||||
|         } | ||||
|  | ||||
|         val distinctFree = free.distinct() | ||||
|         free.clear() | ||||
|         free.addAll(distinctFree) | ||||
|  | ||||
|         removeReservedFromFreePool() | ||||
|         allocateCx16VirtualRegisters() | ||||
|         retainAllowed() | ||||
|     } | ||||
|  | ||||
|     private 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) { | ||||
|             val address = virtualRegistersStart + (2*reg).toUInt() | ||||
|             if(address<=0xffu) { | ||||
|                 allocatedVariables["cx16.r${reg}"]   = VarAllocation(address, DataType.UWORD, 2)       // cx16.r0 .. cx16.r15 | ||||
|                 allocatedVariables["cx16.r${reg}s"]  = VarAllocation(address, DataType.WORD, 2)        // cx16.r0s .. cx16.r15s | ||||
|                 allocatedVariables["cx16.r${reg}L"]  = VarAllocation(address, DataType.UBYTE, 1)       // cx16.r0L .. cx16.r15L | ||||
|                 allocatedVariables["cx16.r${reg}H"]  = VarAllocation(address+1u, DataType.UBYTE, 1)    // cx16.r0H .. cx16.r15H | ||||
|                 allocatedVariables["cx16.r${reg}sL"] = VarAllocation(address, DataType.BYTE, 1)        // cx16.r0sL .. cx16.r15sL | ||||
|                 allocatedVariables["cx16.r${reg}sH"] = VarAllocation(address+1u, DataType.BYTE, 1)     // cx16.r0sH .. cx16.r15sH | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
							
								
								
									
										52
									
								
								codeCore/src/prog8/code/target/zp/PETZeropage.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								codeCore/src/prog8/code/target/zp/PETZeropage.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | ||||
| package prog8.code.target.zp | ||||
|  | ||||
| import prog8.code.core.CompilationOptions | ||||
| import prog8.code.core.InternalCompilerException | ||||
| import prog8.code.core.Zeropage | ||||
| import prog8.code.core.ZeropageType | ||||
|  | ||||
|  | ||||
| // reference: http://www.zimmers.net/cbmpics/cbm/PETx/petmem.txt | ||||
|  | ||||
| class PETZeropage(options: CompilationOptions) : Zeropage(options) { | ||||
|  | ||||
|     override val SCRATCH_B1 = 0xb3u      // temp storage for a single byte | ||||
|     override val SCRATCH_REG = 0xb4u     // temp storage for a register byte, must be B1+1 | ||||
|     override val SCRATCH_W1 = 0xb6u      // temp storage 1 for a word | ||||
|     override val SCRATCH_W2 = 0xb8u      // temp storage 2 for a word | ||||
|     override val SCRATCH_PTR = 0xb1u     // temp storage for a pointer $b1+$b2 | ||||
|  | ||||
|     init { | ||||
|         if (options.floats && options.zeropage !in arrayOf( | ||||
|                 ZeropageType.FLOATSAFE, | ||||
|                 ZeropageType.BASICSAFE, | ||||
|                 ZeropageType.DONTUSE | ||||
|             )) | ||||
|             throw InternalCompilerException("when floats are enabled, zero page type should be 'floatsafe' or 'basicsafe' or 'dontuse'") | ||||
|  | ||||
|         when (options.zeropage) { | ||||
|             ZeropageType.FULL -> { | ||||
|                 free.addAll(0x00u..0xffu) | ||||
|                 free.removeAll(listOf(0x8du, 0x8eu, 0x8fu, 0x97u, 0x98u, 0x99u, 0x9au, 0x9bu, 0x9eu, 0xa7u, 0xa8u, 0xa9u, 0xaau))        // these are updated/used by IRQ | ||||
|             } | ||||
|             ZeropageType.KERNALSAFE -> { | ||||
|                 free.addAll(0x00u..0xffu) | ||||
|                 free.removeAll(listOf(0x8du, 0x8eu, 0x8fu, 0x97u, 0x98u, 0x99u, 0x9au, 0x9bu, 0x9eu, 0xa7u, 0xa8u, 0xa9u, 0xaau))        // these are updated/used by IRQ | ||||
|             } | ||||
|             ZeropageType.FLOATSAFE, | ||||
|             ZeropageType.BASICSAFE -> { | ||||
|                 free.addAll(0xb1u..0xbau)       // TODO more? | ||||
|             } | ||||
|             ZeropageType.DONTUSE -> { | ||||
|                 free.clear()  // don't use zeropage at all | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val distinctFree = free.distinct() | ||||
|         free.clear() | ||||
|         free.addAll(distinctFree) | ||||
|  | ||||
|         removeReservedFromFreePool() | ||||
|         retainAllowed() | ||||
|     } | ||||
| } | ||||
							
								
								
									
										43
									
								
								codeGenCpu6502/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								codeGenCpu6502/build.gradle.kts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,43 @@ | ||||
| plugins { | ||||
|     kotlin("jvm") | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     implementation(project(":codeCore")) | ||||
|     implementation(project(":simpleAst")) | ||||
|     // implementation("org.jetbrains.kotlin:kotlin-stdlib-jdk8") | ||||
|     // implementation "org.jetbrains.kotlin:kotlin-reflect" | ||||
|     implementation("com.michael-bull.kotlin-result:kotlin-result-jvm:2.1.0") | ||||
|  | ||||
|     testImplementation("io.kotest:kotest-runner-junit5-jvm:5.9.1") | ||||
|     testImplementation("io.kotest:kotest-framework-datatest:5.9.1") | ||||
| } | ||||
|  | ||||
| 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") | ||||
|     } | ||||
| } | ||||
							
								
								
									
										19
									
								
								codeGenCpu6502/codeGenCpu6502.iml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								codeGenCpu6502/codeGenCpu6502.iml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| <?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" /> | ||||
|       <sourceFolder url="file://$MODULE_DIR$/test" isTestSource="true" /> | ||||
|       <excludeFolder url="file://$MODULE_DIR$/build" /> | ||||
|     </content> | ||||
|     <orderEntry type="inheritedJdk" /> | ||||
|     <orderEntry type="sourceFolder" forTests="false" /> | ||||
|     <orderEntry type="library" name="KotlinJavaRuntime" level="project" /> | ||||
|     <orderEntry type="module" module-name="codeCore" /> | ||||
|     <orderEntry type="module" module-name="simpleAst" /> | ||||
|     <orderEntry type="library" name="michael.bull.kotlin.result.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> | ||||
| </module> | ||||
							
								
								
									
										2293
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/AsmGen.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2293
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/AsmGen.kt
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										912
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/AsmOptimizer.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										912
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/AsmOptimizer.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,912 @@ | ||||
| package prog8.codegen.cpu6502 | ||||
|  | ||||
| import prog8.code.GENERATED_LABEL_PREFIX | ||||
| import prog8.code.StConstant | ||||
| import prog8.code.StMemVar | ||||
| import prog8.code.SymbolTable | ||||
| import prog8.code.core.ICompilationTarget | ||||
|  | ||||
|  | ||||
| // note: see https://wiki.nesdev.org/w/index.php/6502_assembly_optimisations | ||||
|  | ||||
|  | ||||
| internal fun optimizeAssembly(lines: MutableList<String>, machine: ICompilationTarget, symbolTable: SymbolTable): Int { | ||||
|  | ||||
|     var numberOfOptimizations = 0 | ||||
|  | ||||
|     var linesByFour = getLinesBy(lines, 4) | ||||
|  | ||||
|     var mods = optimizeIncDec(linesByFour) | ||||
|     if(mods.isNotEmpty()) { | ||||
|         apply(mods, lines) | ||||
|         linesByFour = getLinesBy(lines, 4) | ||||
|         numberOfOptimizations++ | ||||
|     } | ||||
|  | ||||
|     mods = optimizeStoreLoadSame(linesByFour, machine, symbolTable) | ||||
|     if(mods.isNotEmpty()) { | ||||
|         apply(mods, lines) | ||||
|         linesByFour = getLinesBy(lines, 4) | ||||
|         numberOfOptimizations++ | ||||
|     } | ||||
|  | ||||
|     mods = optimizeJsrRtsAndOtherCombinations(linesByFour) | ||||
|     if(mods.isNotEmpty()) { | ||||
|         apply(mods, lines) | ||||
|         linesByFour = getLinesBy(lines, 4) | ||||
|         numberOfOptimizations++ | ||||
|     } | ||||
|  | ||||
|     mods = optimizeUselessPushPopStack(linesByFour) | ||||
|     if(mods.isNotEmpty()) { | ||||
|         apply(mods, lines) | ||||
|         linesByFour = getLinesBy(lines, 4) | ||||
|         numberOfOptimizations++ | ||||
|     } | ||||
|  | ||||
|     mods = optimizeUnneededTempvarInAdd(linesByFour) | ||||
|     if(mods.isNotEmpty()) { | ||||
|         apply(mods, lines) | ||||
|         linesByFour = getLinesBy(lines, 4) | ||||
|         numberOfOptimizations++ | ||||
|     } | ||||
|  | ||||
|     mods = optimizeTSBtoRegularOr(linesByFour) | ||||
|     if(mods.isNotEmpty()) { | ||||
|         apply(mods, lines) | ||||
|         linesByFour = getLinesBy(lines, 4) | ||||
|         numberOfOptimizations++ | ||||
|     } | ||||
|  | ||||
|     var linesByFourteen = getLinesBy(lines, 14) | ||||
|     mods = optimizeSameAssignments(linesByFourteen, machine, symbolTable) | ||||
|     if(mods.isNotEmpty()) { | ||||
|         apply(mods, lines) | ||||
|         linesByFourteen = getLinesBy(lines, 14) | ||||
|         numberOfOptimizations++ | ||||
|     } | ||||
|  | ||||
|     mods = optimizeSamePointerIndexingAndUselessBeq(linesByFourteen) | ||||
|     if(mods.isNotEmpty()) { | ||||
|         apply(mods, lines) | ||||
|         linesByFourteen = getLinesBy(lines, 14) | ||||
|         numberOfOptimizations++ | ||||
|     } | ||||
|  | ||||
|     mods = optimizeAddWordToSameVariableOrExtraRegisterLoadInWordStore(linesByFourteen) | ||||
|     if(mods.isNotEmpty()) { | ||||
|         apply(mods, lines) | ||||
|         linesByFourteen = getLinesBy(lines, 14) | ||||
|         numberOfOptimizations++ | ||||
|     } | ||||
|  | ||||
|     return numberOfOptimizations | ||||
| } | ||||
|  | ||||
| private fun String.isBranch() = this.startsWith("b") | ||||
| private fun String.isStoreReg() = this.startsWith("sta") || this.startsWith("sty") || this.startsWith("stx") | ||||
| private fun String.isStoreRegOrZero() = this.isStoreReg() || this.startsWith("stz") | ||||
| private fun String.isLoadReg() = this.startsWith("lda") || this.startsWith("ldy") || this.startsWith("ldx") | ||||
|  | ||||
| private class Modification(val lineIndex: Int, val remove: Boolean, val replacement: String?, val removeLabel: Boolean=false) | ||||
|  | ||||
| private fun apply(modifications: List<Modification>, lines: MutableList<String>) { | ||||
|     for (modification in modifications.sortedBy { it.lineIndex }.reversed()) { | ||||
|         if(modification.remove) { | ||||
|             if(modification.removeLabel) | ||||
|                 lines.removeAt(modification.lineIndex) | ||||
|             else { | ||||
|                 val line = lines[modification.lineIndex] | ||||
|                 if (line.length < 2 || line[0] == ';' || line.trimStart()[0] == ';') | ||||
|                     lines.removeAt(modification.lineIndex) | ||||
|                 else if (haslabel(line)) { | ||||
|                     val label = keeplabel(line) | ||||
|                     if (label.isNotEmpty()) | ||||
|                         lines[modification.lineIndex] = label | ||||
|                     else | ||||
|                         lines.removeAt(modification.lineIndex) | ||||
|                 } else lines.removeAt(modification.lineIndex) | ||||
|             } | ||||
|         } | ||||
|         else | ||||
|             lines[modification.lineIndex] = modification.replacement!! | ||||
|     } | ||||
| } | ||||
|  | ||||
| private fun haslabel(line: String): Boolean { | ||||
|     return line.length>1 && line[0]!=';' && (!line[0].isWhitespace() || ':' in line) | ||||
| } | ||||
|  | ||||
| private fun keeplabel(line: String): String { | ||||
|     if(':' in line) | ||||
|         return line.substringBefore(':') + ':' | ||||
|     val splits = line.split('\t', ' ', limit=2) | ||||
|     return if(splits.size>1) splits[0] + ':' else "" | ||||
| } | ||||
|  | ||||
| private fun getLinesBy(lines: MutableList<String>, windowSize: Int) = | ||||
| // all lines (that aren't empty or comments) in sliding windows of certain size | ||||
|         lines.asSequence().withIndex().filter { it.value.isNotBlank() && !it.value.trimStart().startsWith(';') }.windowed(windowSize, partialWindows = false) | ||||
|  | ||||
| private fun optimizeSameAssignments( | ||||
|     linesByFourteen: Sequence<List<IndexedValue<String>>>, | ||||
|     machine: ICompilationTarget, | ||||
|     symbolTable: SymbolTable | ||||
| ): List<Modification> { | ||||
|  | ||||
|     // Optimize sequential assignments of the same value to various targets (bytes, words, floats) | ||||
|     // the float one is the one that requires 2*7=14 lines of code to check... | ||||
|     // The better place to do this is in the Compiler instead and never create these types of assembly, but hey | ||||
|  | ||||
|     val mods = mutableListOf<Modification>() | ||||
|     for (lines in linesByFourteen) { | ||||
|         val first = lines[0].value.trimStart() | ||||
|         val second = lines[1].value.trimStart() | ||||
|         val third = lines[2].value.trimStart() | ||||
|         val fourth = lines[3].value.trimStart() | ||||
|         val fifth = lines[4].value.trimStart() | ||||
|         val sixth = lines[5].value.trimStart() | ||||
|         val seventh = lines[6].value.trimStart() | ||||
|         val eighth = lines[7].value.trimStart() | ||||
|  | ||||
|         if(first.startsWith("lda") && second.startsWith("ldy") && third.startsWith("sta") && fourth.startsWith("sty") && | ||||
|                 fifth.startsWith("lda") && sixth.startsWith("ldy") && seventh.startsWith("sta") && eighth.startsWith("sty")) { | ||||
|             val firstvalue = first.substring(4) | ||||
|             val secondvalue = second.substring(4) | ||||
|             val thirdvalue = fifth.substring(4) | ||||
|             val fourthvalue = sixth.substring(4) | ||||
|             if(firstvalue==thirdvalue && secondvalue==fourthvalue) { | ||||
|                 // lda/ldy   sta/sty   twice the same word -->  remove second lda/ldy pair (fifth and sixth lines) | ||||
|                 val address1 = getAddressArg(first, symbolTable) | ||||
|                 val address2 = getAddressArg(second, symbolTable) | ||||
|                 if(address1==null || address2==null || (!machine.isIOAddress(address1) && !machine.isIOAddress(address2))) { | ||||
|                     mods.add(Modification(lines[4].index, true, null)) | ||||
|                     mods.add(Modification(lines[5].index, true, null)) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if(first.startsWith("lda") && second.startsWith("sta") && third.startsWith("lda") && fourth.startsWith("sta")) { | ||||
|             val firstvalue = first.substring(4) | ||||
|             val secondvalue = third.substring(4) | ||||
|             if(firstvalue==secondvalue) { | ||||
|                 // lda value / sta ? / lda same-value / sta ?  -> remove second lda (third line) | ||||
|                 val address = getAddressArg(first, symbolTable) | ||||
|                 if(address==null || !machine.isIOAddress(address)) | ||||
|                     mods.add(Modification(lines[2].index, true, null)) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if(first.startsWith("lda") && second.startsWith("ldy") && third.startsWith("sta") && fourth.startsWith("sty") && | ||||
|                 fifth.startsWith("lda") && sixth.startsWith("ldy") && | ||||
|                 (seventh.startsWith("jsr  floats.copy_float") || seventh.startsWith("jsr  cx16flt.copy_float"))) { | ||||
|  | ||||
|             val nineth = lines[8].value.trimStart() | ||||
|             val tenth = lines[9].value.trimStart() | ||||
|             val eleventh = lines[10].value.trimStart() | ||||
|             val twelveth = lines[11].value.trimStart() | ||||
|             val thirteenth = lines[12].value.trimStart() | ||||
|             val fourteenth = lines[13].value.trimStart() | ||||
|  | ||||
|             if(eighth.startsWith("lda") && nineth.startsWith("ldy") && tenth.startsWith("sta") && eleventh.startsWith("sty") && | ||||
|                     twelveth.startsWith("lda") && thirteenth.startsWith("ldy") && | ||||
|                     (fourteenth.startsWith("jsr  floats.copy_float") || fourteenth.startsWith("jsr  cx16flt.copy_float"))) { | ||||
|  | ||||
|                 if(first.substring(4) == eighth.substring(4) && second.substring(4)==nineth.substring(4)) { | ||||
|                     // identical float init | ||||
|                     mods.add(Modification(lines[7].index, true, null)) | ||||
|                     mods.add(Modification(lines[8].index, true, null)) | ||||
|                     mods.add(Modification(lines[9].index, true, null)) | ||||
|                     mods.add(Modification(lines[10].index, true, null)) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         var overlappingMods = false | ||||
|         /* | ||||
|         sta  prog8_lib.retval_intermX      ; remove | ||||
|         sty  prog8_lib.retval_intermY      ; remove | ||||
|         lda  prog8_lib.retval_intermX      ; remove | ||||
|         ldy  prog8_lib.retval_intermY      ; remove | ||||
|         sta  A1 | ||||
|         sty  A2 | ||||
|          */ | ||||
|         if(first.isStoreReg() && second.isStoreReg() | ||||
|             && third.isLoadReg() && fourth.isLoadReg() | ||||
|             && fifth.isStoreReg() && sixth.isStoreReg()) { | ||||
|             val reg1 = first[2] | ||||
|             val reg2 = second[2] | ||||
|             val reg3 = third[2] | ||||
|             val reg4 = fourth[2] | ||||
|             val reg5 = fifth[2] | ||||
|             val reg6 = sixth[2] | ||||
|             if (reg1 == reg3 && reg1 == reg5 && reg2 == reg4 && reg2 == reg6) { | ||||
|                 val firstvalue = first.substring(4) | ||||
|                 val secondvalue = second.substring(4) | ||||
|                 val thirdvalue = third.substring(4) | ||||
|                 val fourthvalue = fourth.substring(4) | ||||
|                 if(firstvalue.contains("prog8_lib.retval_interm") && secondvalue.contains("prog8_lib.retval_interm") | ||||
|                     && firstvalue==thirdvalue && secondvalue==fourthvalue) { | ||||
|                     mods.add(Modification(lines[0].index, true, null)) | ||||
|                     mods.add(Modification(lines[1].index, true, null)) | ||||
|                     mods.add(Modification(lines[2].index, true, null)) | ||||
|                     mods.add(Modification(lines[3].index, true, null)) | ||||
|                     overlappingMods = true | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* | ||||
|         sta  A1 | ||||
|         sty  A2 | ||||
|         lda  A1     ; can be removed | ||||
|         ldy  A2     ; can be removed if not followed by a branch instuction | ||||
|          */ | ||||
|         if(!overlappingMods && first.isStoreReg() && second.isStoreReg() | ||||
|             && third.isLoadReg() && fourth.isLoadReg()) { | ||||
|             val reg1 = first[2] | ||||
|             val reg2 = second[2] | ||||
|             val reg3 = third[2] | ||||
|             val reg4 = fourth[2] | ||||
|             if(reg1==reg3 && reg2==reg4) { | ||||
|                 val firstvalue = first.substring(4) | ||||
|                 val secondvalue = second.substring(4) | ||||
|                 val thirdvalue = third.substring(4) | ||||
|                 val fourthvalue = fourth.substring(4) | ||||
|                 if(firstvalue==thirdvalue && secondvalue == fourthvalue) { | ||||
|                     val address = getAddressArg(first, symbolTable) | ||||
|                     if(address==null || !machine.isIOAddress(address)) { | ||||
|                         overlappingMods = true | ||||
|                         mods.add(Modification(lines[2].index, true, null)) | ||||
|                         if (!fifth.startsWith('b')) | ||||
|                             mods.add(Modification(lines[3].index, true, null)) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* | ||||
|         sta  A1 | ||||
|         sty  A2     ; ... or stz | ||||
|         lda  A1     ; can be removed if not followed by a branch instruction | ||||
|          */ | ||||
|         if(!overlappingMods && first.isStoreReg() && second.isStoreRegOrZero() | ||||
|             && third.isLoadReg() && !fourth.isBranch()) { | ||||
|             val reg1 = first[2] | ||||
|             val reg3 = third[2] | ||||
|             if(reg1==reg3) { | ||||
|                 val firstvalue = first.substring(4) | ||||
|                 val thirdvalue = third.substring(4) | ||||
|                 if(firstvalue==thirdvalue) { | ||||
|                     val address = getAddressArg(first, symbolTable) | ||||
|                     if(address==null || !machine.isIOAddress(address)) { | ||||
|                         overlappingMods = true | ||||
|                         mods.add(Modification(lines[2].index, true, null)) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* | ||||
|         sta  A1 | ||||
|         ldy  A1     ; make tay | ||||
|         sta  A1     ; remove | ||||
|          */ | ||||
|         if(!overlappingMods && first.startsWith("sta") && second.isLoadReg() | ||||
|             && third.startsWith("sta") && second.length>4) { | ||||
|             val firstvalue = first.substring(4) | ||||
|             val secondvalue = second.substring(4) | ||||
|             val thirdvalue = third.substring(4) | ||||
|             if(firstvalue==secondvalue && firstvalue==thirdvalue) { | ||||
|                 val address = getAddressArg(first, symbolTable) | ||||
|                 if(address==null || !machine.isIOAddress(address)) { | ||||
|                     overlappingMods = true | ||||
|                     val reg2 = second[2] | ||||
|                     mods.add(Modification(lines[1].index, false, "  ta$reg2")) | ||||
|                     mods.add(Modification(lines[2].index, true, null)) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* | ||||
|         sta  A   ; or stz     double store, remove this first one | ||||
|         sta  A   ; or stz | ||||
|         However, this cannot be done relyably because 'A' could be a constant symbol referring to an I/O address. | ||||
|         We can't see that here and would otherwise delete valid double stores. | ||||
|         */ | ||||
|     } | ||||
|  | ||||
|     return mods | ||||
| } | ||||
|  | ||||
| private fun optimizeSamePointerIndexingAndUselessBeq(linesByFourteen: Sequence<List<IndexedValue<String>>>): List<Modification> { | ||||
|  | ||||
|     // Optimize same pointer indexing where for instance we load and store to the same ptr index in Y | ||||
|     // if Y isn't modified in between we can omit the second LDY: | ||||
|     //    ldy  #0 | ||||
|     //    lda  (ptr),y | ||||
|     //    ora  #3       ; <-- instruction(s) that don't modify Y | ||||
|     //    ldy  #0       ; <-- can be removed | ||||
|     //    sta  (ptr),y | ||||
|  | ||||
|     val mods = mutableListOf<Modification>() | ||||
|     for (lines in linesByFourteen) { | ||||
|         val first = lines[0].value.trimStart() | ||||
|         val second = lines[1].value.trimStart() | ||||
|         val third = lines[2].value.trimStart() | ||||
|         val fourth = lines[3].value.trimStart() | ||||
|         val fifth = lines[4].value.trimStart() | ||||
|         val sixth = lines[5].value.trimStart() | ||||
|  | ||||
|         if(first.startsWith("ldy") && second.startsWith("lda") && fourth.startsWith("ldy") && fifth.startsWith("sta")) { | ||||
|             val firstvalue = first.substring(4) | ||||
|             val secondvalue = second.substring(4) | ||||
|             val fourthvalue = fourth.substring(4) | ||||
|             val fifthvalue = fifth.substring(4) | ||||
|             if("y" !in third && firstvalue==fourthvalue && secondvalue==fifthvalue && secondvalue.endsWith(",y") && fifthvalue.endsWith(",y")) { | ||||
|                 mods.add(Modification(lines[3].index, true, null)) | ||||
|             } | ||||
|         } | ||||
|         if(first.startsWith("ldy") && second.startsWith("lda") && fifth.startsWith("ldy") && sixth.startsWith("sta")) { | ||||
|             val firstvalue = first.substring(4) | ||||
|             val secondvalue = second.substring(4) | ||||
|             val fifthvalue = fifth.substring(4) | ||||
|             val sixthvalue = sixth.substring(4) | ||||
|             if("y" !in third && "y" !in fourth && firstvalue==fifthvalue && secondvalue==sixthvalue && secondvalue.endsWith(",y") && sixthvalue.endsWith(",y")) { | ||||
|                 mods.add(Modification(lines[4].index, true, null)) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|  | ||||
|         /* | ||||
|     beq  + | ||||
|     lda  #1 | ||||
| + | ||||
| [ optional:  label_xxxx_shortcut   line here] | ||||
|     beq  label_xxxx_shortcut   /  bne label_xxxx_shortcut | ||||
| or *_afterif labels. | ||||
|  | ||||
| This gets generated after certain if conditions, and only the branch instruction is needed in these cases. | ||||
|          */ | ||||
|  | ||||
|         val autoLabelPrefix = GENERATED_LABEL_PREFIX | ||||
|         if(first=="beq  +" && second=="lda  #1" && third=="+") { | ||||
|             if((fourth.startsWith("beq  $autoLabelPrefix") || fourth.startsWith("bne  $autoLabelPrefix")) && | ||||
|                 (fourth.endsWith("_shortcut") || fourth.endsWith("_afterif") || fourth.endsWith("_shortcut:") || fourth.endsWith("_afterif:"))) { | ||||
|                 mods.add(Modification(lines[0].index, true, null)) | ||||
|                 mods.add(Modification(lines[1].index, true, null)) | ||||
|                 mods.add(Modification(lines[2].index, true, null)) | ||||
|             } | ||||
|             else if(fourth.startsWith(autoLabelPrefix) && (fourth.endsWith("_shortcut") || fourth.endsWith("_shortcut:"))) { | ||||
|                 if((fifth.startsWith("beq  $autoLabelPrefix") || fifth.startsWith("bne  $autoLabelPrefix")) && | ||||
|                     (fifth.endsWith("_shortcut") || fifth.endsWith("_afterif") || fifth.endsWith("_shortcut:") || fifth.endsWith("_afterif:"))) { | ||||
|                     mods.add(Modification(lines[0].index, true, null)) | ||||
|                     mods.add(Modification(lines[1].index, true, null)) | ||||
|                     mods.add(Modification(lines[2].index, true, null)) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return mods | ||||
| } | ||||
|  | ||||
| private fun optimizeStoreLoadSame( | ||||
|     linesByFour: Sequence<List<IndexedValue<String>>>, | ||||
|     machine: ICompilationTarget, | ||||
|     symbolTable: SymbolTable | ||||
| ): List<Modification> { | ||||
|     val mods = mutableListOf<Modification>() | ||||
|     for (lines in linesByFour) { | ||||
|         val first = lines[1].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 | ||||
|         if ((first.startsWith("sta ") && second.startsWith("lda ")) || | ||||
|                 (first.startsWith("stx ") && second.startsWith("ldx ")) || | ||||
|                 (first.startsWith("sty ") && second.startsWith("ldy ")) || | ||||
|                 (first.startsWith("lda ") && second.startsWith("lda ")) || | ||||
|                 (first.startsWith("ldy ") && second.startsWith("ldy ")) || | ||||
|                 (first.startsWith("ldx ") && second.startsWith("ldx ")) | ||||
|         ) { | ||||
|             val attemptRemove = | ||||
|                 if(third.isBranch()) { | ||||
|                     // a branch instruction follows, we can only remove the load instruction if | ||||
|                     // another load instruction of the same register precedes the store instruction | ||||
|                     // (otherwise wrong cpu flags are used) | ||||
|                     val loadinstruction = second.take(3) | ||||
|                     lines[0].value.trimStart().startsWith(loadinstruction) | ||||
|                 } | ||||
|                 else { | ||||
|                     // no branch instruction follows, we can remove the load instruction | ||||
|                     val address = getAddressArg(lines[2].value, symbolTable) | ||||
|                     address==null || !machine.isIOAddress(address) | ||||
|                 } | ||||
|  | ||||
|             if(attemptRemove) { | ||||
|                 val firstLoc = first.substring(4).trimStart() | ||||
|                 val secondLoc = second.substring(4).trimStart() | ||||
|                 if (firstLoc == secondLoc) | ||||
|                     mods.add(Modification(lines[2].index, true, null)) | ||||
|             } | ||||
|         } | ||||
|         else if(first=="pha" && second=="pla" || | ||||
|             first=="phx" && second=="plx" || | ||||
|             first=="phy" && second=="ply" || | ||||
|             first=="php" && second=="plp") { | ||||
|             mods.add(Modification(lines[1].index, true, null)) | ||||
|             mods.add(Modification(lines[2].index, true, null)) | ||||
|         } else if(first=="pha" && second=="plx") { | ||||
|             mods.add(Modification(lines[1].index, true, null)) | ||||
|             mods.add(Modification(lines[2].index, false, "  tax")) | ||||
|         } else if(first=="pha" && second=="ply") { | ||||
|             mods.add(Modification(lines[1].index, true, null)) | ||||
|             mods.add(Modification(lines[2].index, false, "  tay")) | ||||
|         } else if(first=="phx" && second=="pla") { | ||||
|             mods.add(Modification(lines[1].index, true, null)) | ||||
|             mods.add(Modification(lines[2].index, false, "  txa")) | ||||
|         } else if(first=="phy" && second=="pla") { | ||||
|             mods.add(Modification(lines[1].index, true, null)) | ||||
|             mods.add(Modification(lines[2].index, false, "  tya")) | ||||
|         } | ||||
|  | ||||
|  | ||||
|         // lda X + sta X,  ldy X + sty X,   ldx X + stx X  -> the second instruction can be eliminated | ||||
|         if (first.startsWith("lda ") && second.startsWith("sta ") || | ||||
|             first.startsWith("ldx ") && second.startsWith("stx ") || | ||||
|             first.startsWith("ldy ") && second.startsWith("sty ") | ||||
|         ) { | ||||
|             val firstLoc = first.substring(4).trimStart() | ||||
|             val secondLoc = second.substring(4).trimStart() | ||||
|             if (firstLoc == secondLoc) | ||||
|                 mods.add(Modification(lines[2].index, true, null)) | ||||
|         } | ||||
|  | ||||
|         //  all 3 registers:  lda VALUE + sta SOMEWHERE + lda VALUE  -> last load can be eliminated | ||||
|         if (first.startsWith("lda ") && second.startsWith("sta ") && third.startsWith("lda ") || | ||||
|             first.startsWith("ldx ") && second.startsWith("stx ") && third.startsWith("ldx ") || | ||||
|             first.startsWith("ldy ") && second.startsWith("sty ") && third.startsWith("ldy ") | ||||
|         ) { | ||||
|             val firstVal = first.substring(4).trimStart() | ||||
|             val thirdVal = third.substring(4).trimStart() | ||||
|             if (firstVal == thirdVal) | ||||
|                 mods.add(Modification(lines[3].index, true, null)) | ||||
|         } | ||||
|     } | ||||
|     return mods | ||||
| } | ||||
|  | ||||
| private val identifierRegex = Regex("""^([a-zA-Z_$][a-zA-Z\d_.$]*)""") | ||||
|  | ||||
| private fun getAddressArg(line: String, symbolTable: SymbolTable): UInt? { | ||||
|     // try to get the constant value address, could return null if it's a symbol instead | ||||
|     val loadArg = line.trimStart().substring(3).trim() | ||||
|     return when { | ||||
|         loadArg.startsWith('$') -> loadArg.substring(1).toUIntOrNull(16) | ||||
|         loadArg.startsWith('%') -> loadArg.substring(1).toUIntOrNull(2) | ||||
|         loadArg.startsWith('#') -> null | ||||
|         loadArg.startsWith('(') -> null | ||||
|         loadArg[0].isLetter() -> { | ||||
|             val identMatch = identifierRegex.find(loadArg) | ||||
|             if(identMatch!=null) { | ||||
|                 val identifier = identMatch.value | ||||
|                 when (val symbol = symbolTable.flat[identifier]) { | ||||
|                     is StConstant -> symbol.value.toUInt() | ||||
|                     is StMemVar -> symbol.address | ||||
|                     else -> null | ||||
|                 } | ||||
|             } else null | ||||
|         } | ||||
|         else -> loadArg.substring(1).toUIntOrNull() | ||||
|     } | ||||
| } | ||||
|  | ||||
| private fun optimizeIncDec(linesByFour: Sequence<List<IndexedValue<String>>>): List<Modification> { | ||||
|     // sometimes, iny+dey / inx+dex / dey+iny / dex+inx sequences are generated, these can be eliminated. | ||||
|     val mods = mutableListOf<Modification>() | ||||
|     for (lines in linesByFour) { | ||||
|         val first = lines[0].value | ||||
|         val second = lines[1].value | ||||
|         if ((" iny" in first || "\tiny" in first) && (" dey" in second || "\tdey" in second) | ||||
|                 || (" inx" in first || "\tinx" in first) && (" dex" in second || "\tdex" in second) | ||||
|                 || (" ina" in first || "\tina" in first) && (" dea" in second || "\tdea" in second) | ||||
|                 || (" inc  a" in first || "\tinc  a" in first) && (" dec  a" in second || "\tdec  a" in second) | ||||
|                 || (" dey" in first || "\tdey" in first) && (" iny" in second || "\tiny" in second) | ||||
|                 || (" dex" in first || "\tdex" in first) && (" inx" in second || "\tinx" in second) | ||||
|                 || (" dea" in first || "\tdea" in first) && (" ina" in second || "\tina" in second) | ||||
|                 || (" dec  a" in first || "\tdec  a" in first) && (" inc  a" in second || "\tinc  a" in second)) { | ||||
|             mods.add(Modification(lines[0].index, true, null)) | ||||
|             mods.add(Modification(lines[1].index, true, null)) | ||||
|         } | ||||
|  | ||||
|     } | ||||
|     return mods | ||||
| } | ||||
|  | ||||
| private fun optimizeJsrRtsAndOtherCombinations(linesByFour: Sequence<List<IndexedValue<String>>>): List<Modification> { | ||||
|     // jsr Sub + rts -> jmp Sub | ||||
|     // jmp Sub + rts -> jmp Sub | ||||
|     // rts + jmp -> remove jmp | ||||
|     // rts + bxx -> remove bxx | ||||
|     // lda  + cmp #0 -> remove cmp,  same for cpy and cpx. | ||||
|     // bra/jmp + bra/jmp -> remove second bra/jmp   (bra bra / jmp jmp are not removed because this is likely a jump table!) | ||||
|     // and some other optimizations. | ||||
|  | ||||
|     val mods = mutableListOf<Modification>() | ||||
|     for (lines in linesByFour) { | ||||
|         val first = lines[0].value | ||||
|         val second = lines[1].value | ||||
|         val third = lines[2].value | ||||
|  | ||||
|         if(!haslabel(second)) { | ||||
|             if ((" jmp" in first || "\tjmp" in first ) && (" rts" in second || "\trts" in second)) { | ||||
|                 mods += Modification(lines[1].index, true, null) | ||||
|             } | ||||
|             else if ((" jsr" in first || "\tjsr" in first ) && (" rts" in second || "\trts" in second)) { | ||||
|                 if("floats.pushFAC" !in first && "floats.popFAC" !in first) {       // these 2 routines depend on being called with JSR!! | ||||
|                     mods += Modification(lines[0].index, false, lines[0].value.replace("jsr", "jmp")) | ||||
|                     mods += Modification(lines[1].index, true, null) | ||||
|                 } | ||||
|             } | ||||
|             else if (" rts" in first || "\trts" in first) { | ||||
|                 if (" jmp" in second || "\tjmp" in second) | ||||
|                     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) | ||||
|                     mods += Modification(lines[1].index, true, null) | ||||
|                 else if (" bcs" in second || "\tbcs" in second) | ||||
|                     mods += Modification(lines[1].index, true, null) | ||||
|                 else if (" beq" in second || "\tbeq" in second) | ||||
|                     mods += Modification(lines[1].index, true, null) | ||||
|                 else if (" bne" in second || "\tbne" in second) | ||||
|                     mods += Modification(lines[1].index, true, null) | ||||
|                 else if (" bmi" in second || "\tbmi" in second) | ||||
|                     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 ((" 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) || | ||||
|                 (" ldy" in first || "\tldy" in first) && (" cpy  #0" in second || "\tcpy  #0" in second) | ||||
|             ) { | ||||
|                 mods.add(Modification(lines[1].index, true, null)) | ||||
|             } | ||||
|             else if(" cmp  #0" in second || "\tcmp  #0" in second) { | ||||
|                 // there are many instructions that modify A and set the bits... | ||||
|                 for(instr in arrayOf("lda", "ora", "and", "eor", "adc", "sbc", "asl", "cmp", "inc  a", "lsr", "pla", "rol", "ror", "txa", "tya")) { | ||||
|                     if(" $instr" in first || "\t$instr" in first) { | ||||
|                         mods.add(Modification(lines[1].index, true, null)) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             // only remove bra followed by jmp or jmp followed by bra | ||||
|             // bra bra or jmp jmp is likely part of a jump table, which should keep all entries! | ||||
|             if((" bra" in first || "\tbra" in first) && (" jmp" in second || "\tjmp" in second)) { | ||||
|                 mods.add(Modification(lines[1].index, true, null)) | ||||
|             } | ||||
|             if((" jmp" in first || "\tjmp" in first) && (" bra" in second || "\tbra" in second)) { | ||||
|                 mods.add(Modification(lines[1].index, true, null)) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         /* | ||||
|     LDA NUM1 | ||||
|     CMP NUM2 | ||||
|     BCC LABEL | ||||
|     BEQ LABEL | ||||
|  | ||||
| (or something similar) which branches to LABEL when NUM1 <= NUM2. (In this case NUM1 and NUM2 are unsigned numbers.) However, consider the following sequence: | ||||
|  | ||||
|     LDA NUM2 | ||||
|     CMP NUM1 | ||||
|     BCS LABEL | ||||
|          */ | ||||
|         val tfirst = first.trimStart() | ||||
|         val tsecond = second.trimStart() | ||||
|         val tthird = lines[2].value.trimStart() | ||||
|         val tfourth = lines[3].value.trimStart() | ||||
|         if(tfirst.startsWith("lda") && tsecond.startsWith("cmp") && tthird.startsWith("bcc") && tfourth.startsWith("beq")) { | ||||
|             val label = tthird.substring(4) | ||||
|             if(label==tfourth.substring(4)) { | ||||
|                 mods += Modification(lines[0].index, false, "  lda  ${tsecond.substring(4)}") | ||||
|                 mods += Modification(lines[1].index, false, "  cmp  ${tfirst.substring(4)}") | ||||
|                 mods += Modification(lines[2].index, false, "  bcs  $label") | ||||
|                 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 | ||||
| } | ||||
|  | ||||
| private fun optimizeUselessPushPopStack(linesByFour: Sequence<List<IndexedValue<String>>>): List<Modification> { | ||||
|     val mods = mutableListOf<Modification>() | ||||
|  | ||||
|     fun optimize(register: Char, lines: List<IndexedValue<String>>) { | ||||
|         if(lines[0].value.trimStart().startsWith("ph$register")) { | ||||
|             if(lines[2].value.trimStart().startsWith("pl$register")) { | ||||
|                 val second = lines[1].value.trimStart().take(6).lowercase() | ||||
|                 if(register!in second | ||||
|                     && !second.startsWith("jsr") | ||||
|                     && !second.startsWith("pl") | ||||
|                     && !second.startsWith("ph")) { | ||||
|                     mods.add(Modification(lines[0].index, true, null)) | ||||
|                     mods.add(Modification(lines[2].index, true, null)) | ||||
|                 } | ||||
|             } | ||||
|             else if (lines[3].value.trimStart().startsWith("pl$register")) { | ||||
|                 val second = lines[1].value.trimStart().take(6).lowercase() | ||||
|                 val third = lines[2].value.trimStart().take(6).lowercase() | ||||
|                 if(register !in second && register !in third | ||||
|                     && !second.startsWith("jsr") && !third.startsWith("jsr") | ||||
|                     && !second.startsWith("pl") && !third.startsWith("pl") | ||||
|                     && !second.startsWith("ph") && !third.startsWith("ph")) { | ||||
|                     mods.add(Modification(lines[0].index, true, null)) | ||||
|                     mods.add(Modification(lines[3].index, true, null)) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     for (lines in linesByFour) { | ||||
|         optimize('a', lines) | ||||
|         optimize('x', lines) | ||||
|         optimize('y', lines) | ||||
|  | ||||
|         val first = lines[0].value.trimStart() | ||||
|         val second = lines[1].value.trimStart() | ||||
|         val third = lines[2].value.trimStart() | ||||
|         val fourth = lines[3].value.trimStart() | ||||
|  | ||||
|         // phy + ldy + pla -> tya + ldy | ||||
|         // phx + ldx + pla -> txa + ldx | ||||
|         // pha + lda + pla -> nop | ||||
|         // pha + tya + tay + pla -> nop | ||||
|         // pha + txa + tax + pla -> nop | ||||
|         when (first) { | ||||
|             "phy" if second.startsWith("ldy ") && third=="pla" -> { | ||||
|                 mods.add(Modification(lines[2].index, true, null)) | ||||
|                 mods.add(Modification(lines[0].index, false, "  tya")) | ||||
|             } | ||||
|             "phx" if second.startsWith("ldx ") && third=="pla" -> { | ||||
|                 mods.add(Modification(lines[2].index, true, null)) | ||||
|                 mods.add(Modification(lines[0].index, false, "  txa")) | ||||
|             } | ||||
|             "pha" if second.startsWith("lda ") && third=="pla" -> { | ||||
|                 mods.add(Modification(lines[0].index, true, null)) | ||||
|                 mods.add(Modification(lines[1].index, true, null)) | ||||
|                 mods.add(Modification(lines[2].index, true, null)) | ||||
|             } | ||||
|             "pha" if ((second=="tya" && third=="tay") || (second=="txa" && third=="tax")) && fourth=="pla" -> { | ||||
|                 mods.add(Modification(lines[0].index, true, null)) | ||||
|                 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 | ||||
| } | ||||
|  | ||||
| private fun optimizeUnneededTempvarInAdd(linesByFour: Sequence<List<IndexedValue<String>>>): List<Modification> { | ||||
|     // sequence:  sta  P8ZP_SCRATCH_XX  / lda  something / clc / adc  P8ZP_SCRATCH_XX | ||||
|     // this can be performed without the scratch variable:  clc  /  adc  something | ||||
|     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() | ||||
|         val fourth = lines[3].value.trimStart() | ||||
|         if(first.startsWith("sta  P8ZP_SCRATCH_") && second.startsWith("lda") && third.startsWith("clc") && fourth.startsWith("adc  P8ZP_SCRATCH_") ) { | ||||
|             if(fourth.substring(4)==first.substring(4)) { | ||||
|                 mods.add(Modification(lines[0].index, false, "  clc")) | ||||
|                 mods.add(Modification(lines[1].index, false, "  adc  ${second.substring(3).trimStart()}")) | ||||
|                 mods.add(Modification(lines[2].index, true, null)) | ||||
|                 mods.add(Modification(lines[3].index, true, null)) | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return mods | ||||
| } | ||||
|  | ||||
| private fun optimizeAddWordToSameVariableOrExtraRegisterLoadInWordStore(linesByFourteen: Sequence<List<IndexedValue<String>>>): List<Modification> { | ||||
|     /* | ||||
|         ; FIRST SEQUYENCE: P8ZP_SCRATCH_PTR += AY : | ||||
|         clc | ||||
|         adc  P8ZP_SCRATCH_PTR | ||||
|         pha | ||||
|         tya | ||||
|         adc  P8ZP_SCRATCH_PTR+1 | ||||
|         tay | ||||
|         pla | ||||
|         sta  P8ZP_SCRATCH_PTR | ||||
|         sty  P8ZP_SCRATCH_PTR+1 | ||||
|  | ||||
| 	    -> | ||||
|  | ||||
|         clc | ||||
|     	adc  P8ZP_SCRATCH_PTR | ||||
|     	sta  P8ZP_SCRATCH_PTR | ||||
|     	tya | ||||
|     	adc  P8ZP_SCRATCH_PTR+1 | ||||
|     	sta  P8ZP_SCRATCH_PTR+1 | ||||
|  | ||||
|  | ||||
|     	also SECOND SEQUENCE: | ||||
|  | ||||
|         ldx  VALUE/  ldy  VALUE | ||||
| 	    sta  SOMEWHERE_WITHOUT_,x_OR_,y | ||||
| 	    txa /   tya | ||||
|     	ldy  #1 | ||||
|     	sta  SOMEWHERE | ||||
|      	 --> | ||||
|     	sta  SOMEWHERE_WITHOUT_,x_OR_,y | ||||
|     	lda  VALUE | ||||
|     	ldy  #1 | ||||
|     	sta  SOMEWHERE | ||||
|  | ||||
|  | ||||
|         also THIRD SEQUENCE: | ||||
|  | ||||
|         ldx  VALUE | ||||
| 	    ldy  #0 | ||||
| 	    sta  SOMEWHERE_WITHOUT_,x | ||||
| 	    txa | ||||
| 	    iny | ||||
| 	    sta  SOMEWHERE | ||||
|          --> | ||||
| 	    ldy  #0 | ||||
| 	    sta  SOMEWHERE_WITHOUT_,x | ||||
| 	    lda  VALUE | ||||
| 	    iny | ||||
| 	    sta  SOMEWHERE | ||||
|     */ | ||||
|     val mods = mutableListOf<Modification>() | ||||
|     for (lines in linesByFourteen) { | ||||
|         val first = lines[0].value.trimStart() | ||||
|         val second = lines[1].value.trimStart() | ||||
|         val third = lines[2].value.trimStart() | ||||
|         val fourth = lines[3].value.trimStart() | ||||
|         val fifth = lines[4].value.trimStart() | ||||
|         val sixth = lines[5].value.trimStart() | ||||
|         val seventh = lines[6].value.trimStart() | ||||
|         val eight = lines[7].value.trimStart() | ||||
|         val ninth = lines[8].value.trimStart() | ||||
|  | ||||
|         // FIRST SEQUENCE | ||||
|         if(first=="clc" && second.startsWith("adc") && third=="pha" && fourth=="tya" && | ||||
|             fifth.startsWith("adc") && sixth=="tay" && seventh=="pla" && eight.startsWith("sta") && ninth.startsWith("sty")) { | ||||
|             val var2 = second.substring(4) | ||||
|             val var5 = fifth.substring(4).substringBefore('+') | ||||
|             val var8 = eight.substring(4) | ||||
|             val var9 = ninth.substring(4).substringBefore('+') | ||||
|             if(var2==var5 && var2==var8 && var2==var9) { | ||||
|                 if(fifth.endsWith("$var5+1") && ninth.endsWith("$var9+1")) { | ||||
|                     mods.add(Modification(lines[2].index, false, "  sta  $var2")) | ||||
|                     mods.add(Modification(lines[5].index, false, "  sta  $var2+1")) | ||||
|                     mods.add(Modification(lines[6].index, true, null)) | ||||
|                     mods.add(Modification(lines[7].index, true, null)) | ||||
|                     mods.add(Modification(lines[8].index, true, null)) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // SECOND SEQUENCE | ||||
|         if(first.startsWith("ldx ") && second.startsWith("sta ") && | ||||
|             third=="txa" && fourth.startsWith("ldy ") && fifth.startsWith("sta ") | ||||
|         ) { | ||||
|             if(",x" !in second) { | ||||
|                 val value = first.substring(4) | ||||
|                 mods.add(Modification(lines[0].index, true, null)) | ||||
|                 mods.add(Modification(lines[2].index, false, "  lda  $value")) | ||||
|             } | ||||
|         } | ||||
|         if(first.startsWith("ldy ") && second.startsWith("sta ") && | ||||
|             third=="tya" && fourth.startsWith("ldy ") && fifth.startsWith("sta ") | ||||
|         ) { | ||||
|             if(",y" !in second) { | ||||
|                 val value = first.substring(4) | ||||
|                 mods.add(Modification(lines[0].index, true, null)) | ||||
|                 mods.add(Modification(lines[2].index, false, "  lda  $value")) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // THIRD SEQUENCE | ||||
|         if(first.startsWith("ldx ") && second.startsWith("ldy ") && third.startsWith("sta ") && | ||||
|             fourth=="txa" && fifth=="iny" && sixth.startsWith("sta ") | ||||
|         ) { | ||||
|             if(",x" !in third) { | ||||
|                 val value = first.substring(4) | ||||
|                 mods.add(Modification(lines[0].index, true, null)) | ||||
|                 mods.add(Modification(lines[3].index, false, "  lda  $value")) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|     } | ||||
|     return mods | ||||
| } | ||||
							
								
								
									
										32
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/AsmsubHelpers.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/AsmsubHelpers.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| package prog8.codegen.cpu6502 | ||||
|  | ||||
| import prog8.code.ast.PtAsmSub | ||||
| import prog8.code.core.Cx16VirtualRegisters | ||||
| import prog8.code.core.RegisterOrPair | ||||
|  | ||||
|  | ||||
| fun asmsub6502ArgsEvalOrder(sub: PtAsmSub): List<Int> { | ||||
|     val order = mutableListOf<Int>() | ||||
|     // order is: | ||||
|     //  1) cx16 virtual word registers, | ||||
|     //  2) paired CPU registers, | ||||
|     //  3) single CPU registers (order Y,X,A), | ||||
|     //  4) floating point registers (FAC1, FAC2), | ||||
|     //  5) CPU Carry status flag | ||||
|     val args = sub.parameters.withIndex() | ||||
|     val (cx16regs, args2) = args.partition { it.value.first.registerOrPair in Cx16VirtualRegisters } | ||||
|     val pairedRegisters = arrayOf(RegisterOrPair.AX, RegisterOrPair.AY, RegisterOrPair.XY) | ||||
|     val (pairedRegs , args3) = args2.partition { it.value.first.registerOrPair in pairedRegisters } | ||||
|     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 } | ||||
|     pairedRegs.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}) | ||||
|     floatRegs.forEach { order += it.index } | ||||
|     rest.forEach { order += it.index } | ||||
|     require(order.size==sub.parameters.size) | ||||
|  | ||||
|     return order | ||||
| } | ||||
							
								
								
									
										172
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/AssemblyProgram.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										172
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/AssemblyProgram.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,172 @@ | ||||
| package prog8.codegen.cpu6502 | ||||
|  | ||||
| import prog8.code.GENERATED_LABEL_PREFIX | ||||
| import prog8.code.IAssemblyProgram | ||||
| import prog8.code.core.CompilationOptions | ||||
| import prog8.code.core.ICompilationTarget | ||||
| import prog8.code.core.IErrorReporter | ||||
| import prog8.code.core.OutputType | ||||
| import prog8.code.target.C128Target | ||||
| import prog8.code.target.C64Target | ||||
| import prog8.code.target.PETTarget | ||||
| import java.nio.file.Path | ||||
|  | ||||
|  | ||||
| internal class AssemblyProgram( | ||||
|         override val name: String, | ||||
|         outputDir: Path, | ||||
|         private val compTarget: ICompilationTarget) : IAssemblyProgram { | ||||
|  | ||||
|     private val assemblyFile = outputDir.resolve("$name.asm") | ||||
|     private val prgFile = outputDir.resolve("$name.prg")        // CBM prg executable program | ||||
|     private val xexFile = outputDir.resolve("$name.xex")        // Atari xex executable program | ||||
|     private val binFile = outputDir.resolve("$name.bin") | ||||
|     private val viceMonListFile = outputDir.resolve(C64Target.viceMonListName(name)) | ||||
|     private val listFile = outputDir.resolve("$name.list") | ||||
|  | ||||
|     override fun assemble(options: CompilationOptions, errors: IErrorReporter): Boolean { | ||||
|  | ||||
|         val assemblerCommand: List<String> | ||||
|  | ||||
|         fun addRemainingOptions(command: MutableList<String>, program: Path, assembly: Path): List<String> { | ||||
|             if(options.compTarget.additionalAssemblerOptions.isNotEmpty()) | ||||
|                 command.addAll(options.compTarget.additionalAssemblerOptions) | ||||
|  | ||||
|             command.addAll(listOf("--output", program.toString(), assembly.toString())) | ||||
|             return command | ||||
|         } | ||||
|  | ||||
|         when(options.output) { | ||||
|             OutputType.PRG -> { | ||||
|                 // CBM machines .prg generation. | ||||
|  | ||||
|                 val command = mutableListOf("64tass", "--cbm-prg", "--ascii", "--case-sensitive", "--long-branch", | ||||
|                     "-Wall", "-Wno-implied-reg", "--no-monitor", "--dump-labels", "--vice-labels", "--labels=$viceMonListFile") | ||||
|  | ||||
|                 if(options.warnSymbolShadowing) | ||||
|                     command.add("-Wshadow") | ||||
|                 else | ||||
|                     command.add("-Wno-shadow") | ||||
|  | ||||
|                 if(options.asmQuiet) | ||||
|                     command.add("--quiet") | ||||
|  | ||||
|                 if(options.asmListfile) { | ||||
|                     command.add("--list=$listFile") | ||||
|                 } | ||||
|  | ||||
|                 assemblerCommand = addRemainingOptions(command, prgFile, assemblyFile) | ||||
|                 if(!options.quiet) | ||||
|                     println("\nCreating prg for target ${compTarget.name}.") | ||||
|             } | ||||
|             OutputType.XEX -> { | ||||
|                 // Atari800XL .xex generation. | ||||
|  | ||||
|                 val command = mutableListOf("64tass", "--atari-xex", "--case-sensitive", "--long-branch", | ||||
|                     "-Wall", "-Wno-implied-reg", "--no-monitor", "--dump-labels", "--vice-labels", "--labels=$viceMonListFile") | ||||
|  | ||||
|                 if(options.warnSymbolShadowing) | ||||
|                     command.add("-Wshadow") | ||||
|                 else | ||||
|                     command.add("-Wno-shadow") | ||||
|  | ||||
|                 if(options.asmQuiet) | ||||
|                     command.add("--quiet") | ||||
|  | ||||
|                 if(options.asmListfile) | ||||
|                     command.add("--list=$listFile") | ||||
|  | ||||
|                 assemblerCommand = addRemainingOptions(command,xexFile, assemblyFile) | ||||
|                 if(!options.quiet) | ||||
|                     println("\nCreating xex for target ${compTarget.name}.") | ||||
|             } | ||||
|             OutputType.RAW -> { | ||||
|                 // Neo6502/headerless raw program generation. | ||||
|                 val command = mutableListOf("64tass", "--nostart", "--case-sensitive", "--long-branch", | ||||
|                     "-Wall", "-Wno-implied-reg", "--no-monitor", "--dump-labels", "--vice-labels", "--labels=$viceMonListFile") | ||||
|  | ||||
|                 if(options.warnSymbolShadowing) | ||||
|                     command.add("-Wshadow") | ||||
|                 else | ||||
|                     command.add("-Wno-shadow") | ||||
|  | ||||
|                 if(options.asmQuiet) | ||||
|                     command.add("--quiet") | ||||
|  | ||||
|                 if(options.asmListfile) | ||||
|                     command.add("--list=$listFile") | ||||
|  | ||||
|                 assemblerCommand = addRemainingOptions(command, binFile, assemblyFile) | ||||
|                 if(!options.quiet) | ||||
|                     println("\nCreating raw binary for target ${compTarget.name}.") | ||||
|             } | ||||
|             OutputType.LIBRARY -> { | ||||
|                 // CBM machines library (.bin) generation (with or without 2 byte load address header depending on the compilation target machine) | ||||
|  | ||||
|                 val command = mutableListOf("64tass", "--ascii", "--case-sensitive", "--long-branch", | ||||
|                     "-Wall", "-Wno-implied-reg", "--no-monitor", "--dump-labels", "--vice-labels", "--labels=$viceMonListFile") | ||||
|  | ||||
|                 if(options.warnSymbolShadowing) | ||||
|                     command.add("-Wshadow") | ||||
|                 else | ||||
|                     command.add("-Wno-shadow") | ||||
|  | ||||
|                 if(options.asmQuiet) | ||||
|                     command.add("--quiet") | ||||
|  | ||||
|                 if(options.asmListfile) | ||||
|                     command.add("--list=$listFile") | ||||
|  | ||||
|                 if(compTarget.name in listOf(C64Target.NAME, C128Target.NAME, PETTarget.NAME)) { | ||||
|                     if(!options.quiet) | ||||
|                         println("\nCreating binary library file with header for target ${compTarget.name}.") | ||||
|                     command.add("--cbm-prg") | ||||
|                 } else { | ||||
|                     if(!options.quiet) | ||||
|                         println("\nCreating binary library file without header for target ${compTarget.name}.") | ||||
|                     command.add("--nostart")       // should be headerless bin, because basic has problems doing a normal LOAD"lib",8,1 - need to use BLOAD | ||||
|                 } | ||||
|  | ||||
|                 assemblerCommand = addRemainingOptions(command, binFile, assemblyFile) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val proc = ProcessBuilder(assemblerCommand) | ||||
|         if(!options.quiet) | ||||
|             proc.inheritIO() | ||||
|         val process = proc.start() | ||||
|         val result = process.waitFor() | ||||
|         if (result == 0) { | ||||
|             removeGeneratedLabelsFromMonlist() | ||||
|             generateBreakpointList() | ||||
|         } | ||||
|         return result==0 | ||||
|     } | ||||
|  | ||||
|     private fun removeGeneratedLabelsFromMonlist() { | ||||
|         val pattern = Regex("""al (\w+) \S+$GENERATED_LABEL_PREFIX.+?""") | ||||
|         val lines = viceMonListFile.toFile().readLines() | ||||
|         viceMonListFile.toFile().outputStream().bufferedWriter().use { | ||||
|             for (line in lines) { | ||||
|                 if(pattern.matchEntire(line)==null) | ||||
|                     it.write(line+"\n") | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun generateBreakpointList() { | ||||
|         // builds list of breakpoints, appends to monitor list file | ||||
|         val breakpoints = mutableListOf<String>() | ||||
|         val pattern = Regex("""al (\w+) \S+_prog8_breakpoint_\d+.?""")      // gather breakpoints by the source label that's generated for them | ||||
|         for (line in viceMonListFile.toFile().readLines()) { | ||||
|             val match = pattern.matchEntire(line) | ||||
|             if (match != null) | ||||
|                 breakpoints.add("break $" + match.groupValues[1]) | ||||
|         } | ||||
|         val num = breakpoints.size | ||||
|         breakpoints.add(0, "; breakpoint list now follows") | ||||
|         breakpoints.add(1, "; $num breakpoints have been defined") | ||||
|         breakpoints.add(2, "del") | ||||
|         viceMonListFile.toFile().appendText(breakpoints.joinToString("\n") + "\n") | ||||
|     } | ||||
| } | ||||
							
								
								
									
										1454
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/BuiltinFunctionsAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1454
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/BuiltinFunctionsAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										935
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/ForLoopsAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										935
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/ForLoopsAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,935 @@ | ||||
| package prog8.codegen.cpu6502 | ||||
|  | ||||
| import com.github.michaelbull.result.fold | ||||
| import prog8.code.StMemVar | ||||
| import prog8.code.StStaticVariable | ||||
| import prog8.code.ast.PtForLoop | ||||
| import prog8.code.ast.PtIdentifier | ||||
| import prog8.code.ast.PtRange | ||||
| import prog8.code.core.* | ||||
| import kotlin.math.absoluteValue | ||||
|  | ||||
| internal class ForLoopsAsmGen( | ||||
|     private val asmgen: AsmGen6502Internal, | ||||
|     private val zeropage: Zeropage | ||||
| ) { | ||||
|  | ||||
|     internal fun translate(stmt: PtForLoop) { | ||||
|         val iterableDt = stmt.iterable.type | ||||
|         when(stmt.iterable) { | ||||
|             is PtRange -> { | ||||
|                 val range = (stmt.iterable as PtRange).toConstantIntegerRange() | ||||
|                 if(range==null) { | ||||
|                     translateForOverNonconstRange(stmt, iterableDt, stmt.iterable as PtRange) | ||||
|                 } else { | ||||
|                     translateForOverConstRange(stmt, iterableDt, range) | ||||
|                 } | ||||
|             } | ||||
|             is PtIdentifier -> { | ||||
|                 translateForOverIterableVar(stmt, iterableDt, stmt.iterable as PtIdentifier) | ||||
|             } | ||||
|             else -> throw AssemblyError("can't iterate over ${stmt.iterable.javaClass} - should have been replaced by a variable") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun translateForOverNonconstRange(stmt: PtForLoop, iterableDt: DataType, range: PtRange) { | ||||
|         if(range.step.asConstInteger()!! < -1) { | ||||
|             val limit = range.to.asConstInteger() | ||||
|             if(limit==0) | ||||
|                 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.isByteArray -> forOverNonconstByteRange(stmt, iterableDt, range) | ||||
|             iterableDt.isWordArray && !iterableDt.isSplitWordArray -> forOverNonconstWordRange(stmt, iterableDt, range) | ||||
|             else -> throw AssemblyError("range expression can only be byte or word") | ||||
|         } | ||||
|  | ||||
|         asmgen.loopEndLabels.removeLast() | ||||
|     } | ||||
|  | ||||
|     private fun forOverNonconstByteRange(stmt: PtForLoop, iterableDt: DataType, range: PtRange) { | ||||
|         val stepsize = range.step.asConstInteger()!! | ||||
|         val loopLabel = asmgen.makeLabel("for_loop") | ||||
|         val endLabel = asmgen.makeLabel("for_end") | ||||
|         asmgen.loopEndLabels.add(endLabel) | ||||
|         val varname = asmgen.asmVariableName(stmt.variable) | ||||
|  | ||||
|         asmgen.assignExpressionToVariable(range.from, varname, iterableDt.elementType()) | ||||
|         when (stepsize) { | ||||
|             -1 if 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""") | ||||
|             } | ||||
|             -1 if range.to.asConstInteger()==1 -> { | ||||
|                 // simple loop downto 1 step -1 | ||||
|                 asmgen.out(loopLabel) | ||||
|                 asmgen.translate(stmt.statements) | ||||
|                 asmgen.out(""" | ||||
|                     dec  $varname | ||||
|                     bne  $loopLabel""") | ||||
|             } | ||||
|             1, -1 -> forOverBytesRangeStepOne(range, varname, iterableDt, loopLabel, endLabel, stmt) | ||||
|             else -> forOverBytesRangeStepGreaterOne(range, varname, iterableDt, loopLabel, endLabel, stmt) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun forOverBytesRangeStepOne(range: PtRange, varname: String, iterableDt: DataType, loopLabel: String, endLabel: String, forloop: PtForLoop) { | ||||
|         // bytes range, step 1 or -1 | ||||
|  | ||||
|         val stepsize = range.step.asConstInteger()!! | ||||
|         val incdec = if(stepsize==1) "inc" else "dec" | ||||
|  | ||||
|         if(asmgen.options.romable) { | ||||
|             // cannot use self-modifying code, cannot use cpu stack (because loop can be interrupted halfway) | ||||
|             // so we need to store the loop end value in a newly allocated temporary variable | ||||
|             val toValueVar = asmgen.createTempVarReused(iterableDt.elementType().base, false, range) | ||||
|             asmgen.assignExpressionToRegister(range.to, RegisterOrPair.A) | ||||
|             asmgen.out("  sta  $toValueVar") | ||||
|             // pre-check for end already reached | ||||
|             if(iterableDt.isSignedByteArray) { | ||||
|                 if(stepsize<0) { | ||||
|                     asmgen.out(""" | ||||
|                         clc | ||||
|                         sbc  $varname | ||||
|                         bvc  + | ||||
|                         eor  #$80 | ||||
| +                       bpl  $endLabel""") | ||||
|                 } | ||||
|                 else { | ||||
|                     asmgen.out(""" | ||||
|                         sec | ||||
|                         sbc  $varname | ||||
|                         bvc  + | ||||
|                         eor  #$80 | ||||
| +                       bmi  $endLabel""") | ||||
|                 } | ||||
|             } else { | ||||
|                 if(stepsize<0) { | ||||
|                     asmgen.out(""" | ||||
|                         cmp  $varname | ||||
|                         beq  + | ||||
|                         bcs  $endLabel | ||||
| +""") | ||||
|                 } | ||||
|                 else { | ||||
|                     asmgen.out("  cmp  $varname  |  bcc  $endLabel") | ||||
|                 } | ||||
|             } | ||||
|             asmgen.out(loopLabel) | ||||
|             asmgen.translate(forloop.statements) | ||||
|             asmgen.out(""" | ||||
|                 lda  $varname | ||||
|                 cmp  $toValueVar | ||||
|                 beq  $endLabel | ||||
|                 $incdec  $varname""") | ||||
|             asmgen.jmp(loopLabel) | ||||
|             asmgen.out(endLabel) | ||||
|  | ||||
|         } else { | ||||
|  | ||||
|             // use self-modifying code to store the loop end comparison value | ||||
|             val modifiedLabel = asmgen.makeLabel("for_modified") | ||||
|             asmgen.assignExpressionToRegister(range.to, RegisterOrPair.A) | ||||
|             // pre-check for end already reached | ||||
|             if(iterableDt.isSignedByteArray) { | ||||
|                 asmgen.out("  sta  $modifiedLabel+1") | ||||
|                 if(stepsize<0) { | ||||
|                     asmgen.out(""" | ||||
|                         clc | ||||
|                         sbc  $varname | ||||
|                         bvc  + | ||||
|                         eor  #$80 | ||||
| +                       bpl  $endLabel""") | ||||
|                 } | ||||
|                 else | ||||
|                     asmgen.out(""" | ||||
|                         sec | ||||
|                         sbc  $varname | ||||
|                         bvc  + | ||||
|                         eor  #$80 | ||||
| +                       bmi  $endLabel""") | ||||
|             } else { | ||||
|                 if(stepsize<0) { | ||||
|                     asmgen.out(""" | ||||
|                         cmp  $varname | ||||
|                         beq  + | ||||
|                         bcs  $endLabel | ||||
| +""") | ||||
|                 } | ||||
|                 else { | ||||
|                     asmgen.out("  cmp  $varname  |  bcc  $endLabel") | ||||
|                 } | ||||
|                 asmgen.out("  sta  $modifiedLabel+1") | ||||
|             } | ||||
|             asmgen.out(loopLabel) | ||||
|             asmgen.translate(forloop.statements) | ||||
|             asmgen.out(""" | ||||
|                 lda  $varname | ||||
| $modifiedLabel  cmp  #0         ; modified  | ||||
|                 beq  $endLabel | ||||
|                 $incdec  $varname""") | ||||
|             asmgen.jmp(loopLabel) | ||||
|             asmgen.out(endLabel) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun forOverBytesRangeStepGreaterOne(range: PtRange, varname: String, iterableDt: DataType, loopLabel: String, endLabel: String, forloop: PtForLoop) { | ||||
|         // bytes range, step >= 2 or <= -2 | ||||
|  | ||||
|         val stepsize = range.step.asConstInteger()!! | ||||
|  | ||||
|         val modifiedLabel = asmgen.makeLabel("for_modified") | ||||
|         asmgen.assignExpressionToRegister(range.to, RegisterOrPair.A) | ||||
|         // pre-check for end already reached | ||||
|         if(iterableDt.isSignedByteArray) { | ||||
|             asmgen.out("  sta  $modifiedLabel+1") | ||||
|             if(stepsize<0) | ||||
|                 asmgen.out(""" | ||||
|                     clc | ||||
|                     sbc  $varname | ||||
|                     bvc  + | ||||
|                     eor  #$80 | ||||
| +                   bpl  $endLabel""") | ||||
|             else | ||||
|                 asmgen.out(""" | ||||
|                     sec | ||||
|                     sbc  $varname | ||||
|                     bvc  + | ||||
|                     eor  #$80 | ||||
| +                   bmi  $endLabel""") | ||||
|         } else { | ||||
|             if(stepsize<0) | ||||
|                 asmgen.out(""" | ||||
|                     cmp  $varname | ||||
|                     beq  + | ||||
|                     bcs  $endLabel | ||||
| +""") | ||||
|             else { | ||||
|                 asmgen.out("  cmp  $varname  |  bcc  $endLabel") | ||||
|             } | ||||
|             asmgen.out("  sta  $modifiedLabel+1") | ||||
|         } | ||||
|         asmgen.out(loopLabel) | ||||
|         asmgen.translate(forloop.statements) | ||||
|  | ||||
|         asmgen.romableError("self-modifying code (forloop over bytes range)", forloop.position)  // TODO fix romable; there is self-modifying code below | ||||
|         if(stepsize>0) { | ||||
|             asmgen.out(""" | ||||
|                 lda  $varname | ||||
|                 clc | ||||
|                 adc  #$stepsize | ||||
|                 sta  $varname | ||||
| $modifiedLabel  cmp  #0    ; modified | ||||
|                 bmi  $loopLabel | ||||
|                 beq  $loopLabel""") | ||||
|         } else { | ||||
|             asmgen.out(""" | ||||
|                 lda  $varname | ||||
|                 sec | ||||
|                 sbc  #${stepsize.absoluteValue} | ||||
|                 sta  $varname | ||||
| $modifiedLabel  cmp  #0     ; modified | ||||
|                 bpl  $loopLabel""") | ||||
|         } | ||||
|         asmgen.out(endLabel) | ||||
|     } | ||||
|  | ||||
|     private fun forOverNonconstWordRange(stmt: PtForLoop, iterableDt: DataType, range: PtRange) { | ||||
|         val stepsize = range.step.asConstInteger()!! | ||||
|         val loopLabel = asmgen.makeLabel("for_loop") | ||||
|         val endLabel = asmgen.makeLabel("for_end") | ||||
|         asmgen.loopEndLabels.add(endLabel) | ||||
|         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) | ||||
|             forOverWordsRangeStepOne(range, varname, iterableDt, loopLabel, endLabel, stmt) | ||||
|         else if (stepsize > 0) | ||||
|             forOverWordsRangeStepGreaterOne(range, varname, iterableDt, loopLabel, endLabel, stmt) | ||||
|         else | ||||
|             forOverWordsRangeStepGreaterOneDescending(range, varname, iterableDt, loopLabel, endLabel, stmt) | ||||
|     } | ||||
|  | ||||
|     private fun forOverWordsRangeStepOne(range: PtRange, varname: String, iterableDt: DataType, loopLabel: String, endLabel: String, forloop: PtForLoop) { | ||||
|         // words range, step 1 or -1 | ||||
|         val stepsize = range.step.asConstInteger()!! | ||||
|  | ||||
|         if(asmgen.options.romable) { | ||||
|             // cannot use self-modifying code, cannot use cpu stack (because loop can be interrupted halfway) | ||||
|             // so we need to store the loop end value in a newly allocated temporary variable | ||||
|             val toValueVar = asmgen.createTempVarReused(iterableDt.elementType().base, false, range) | ||||
|             asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY) | ||||
|             precheckFromToWord(iterableDt, stepsize, varname, endLabel) | ||||
|             asmgen.out("  sta  $toValueVar") | ||||
|             asmgen.out("  sty  $toValueVar+1") | ||||
|             asmgen.out(loopLabel) | ||||
|             asmgen.translate(forloop.statements) | ||||
|             asmgen.out(""" | ||||
|                 lda  $varname+1 | ||||
|                 cmp  $toValueVar+1 | ||||
|                 bne  + | ||||
|                 lda  $varname | ||||
|                 cmp  $toValueVar | ||||
|                 beq  $endLabel""") | ||||
|             if(stepsize==1) { | ||||
|                 asmgen.out(""" | ||||
| +                   inc  $varname | ||||
|                     bne  $loopLabel | ||||
|                     inc  $varname+1""") | ||||
|                 asmgen.jmp(loopLabel) | ||||
|             } else { | ||||
|                 asmgen.out(""" | ||||
| +                   lda  $varname | ||||
|                     bne  + | ||||
|                     dec  $varname+1 | ||||
| +                   dec  $varname""") | ||||
|                 asmgen.jmp(loopLabel) | ||||
|             } | ||||
|             asmgen.out(endLabel) | ||||
|  | ||||
|         } else { | ||||
|  | ||||
|             val modifiedLabel = asmgen.makeLabel("for_modified") | ||||
|             val modifiedLabel2 = asmgen.makeLabel("for_modifiedb") | ||||
|             asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY) | ||||
|             precheckFromToWord(iterableDt, stepsize, varname, endLabel) | ||||
|             asmgen.out(""" | ||||
|                 sty  $modifiedLabel+1 | ||||
|                 sta  $modifiedLabel2+1 | ||||
| $loopLabel""") | ||||
|             asmgen.translate(forloop.statements) | ||||
|             asmgen.out(""" | ||||
|                 lda  $varname+1 | ||||
| $modifiedLabel  cmp  #0    ; modified  | ||||
|                 bne  + | ||||
|                 lda  $varname | ||||
| $modifiedLabel2 cmp  #0    ; modified  | ||||
|                 beq  $endLabel""") | ||||
|             if(stepsize==1) { | ||||
|                 asmgen.out(""" | ||||
| +                   inc  $varname | ||||
|                     bne  $loopLabel | ||||
|                     inc  $varname+1""") | ||||
|                 asmgen.jmp(loopLabel) | ||||
|             } else { | ||||
|                 asmgen.out(""" | ||||
| +                   lda  $varname | ||||
|                     bne  + | ||||
|                     dec  $varname+1 | ||||
| +                   dec  $varname""") | ||||
|                 asmgen.jmp(loopLabel) | ||||
|             } | ||||
|             asmgen.out(endLabel) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun forOverWordsRangeStepGreaterOne(range: PtRange, varname: String, iterableDt: DataType, loopLabel: String, endLabel: String, stmt: PtForLoop) { | ||||
|         // (u)words, step >= 2 | ||||
|         val stepsize = range.step.asConstInteger()!! | ||||
|         val modifiedLabel = asmgen.makeLabel("for_modified") | ||||
|         val modifiedLabel2 = asmgen.makeLabel("for_modifiedb") | ||||
|         asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY) | ||||
|         precheckFromToWord(iterableDt, stepsize, varname, endLabel) | ||||
|         asmgen.out(""" | ||||
|                 sty  $modifiedLabel+1 | ||||
|                 sta  $modifiedLabel2+1 | ||||
| $loopLabel""") | ||||
|         asmgen.translate(stmt.statements) | ||||
|  | ||||
|         asmgen.romableError("self-modifying code (forloop over word range)", stmt.position)  // TODO fix romable; there is self-modifying code below | ||||
|         if (iterableDt.isUnsignedWordArray) { | ||||
|             asmgen.out(""" | ||||
|                 lda  $varname | ||||
|                 clc | ||||
|                 adc  #<$stepsize | ||||
|                 sta  $varname | ||||
|                 lda  $varname+1 | ||||
|                 adc  #>$stepsize | ||||
|                 sta  $varname+1 | ||||
| $modifiedLabel  cmp  #0     ; modified | ||||
|                 bcc  $loopLabel | ||||
|                 bne  $endLabel | ||||
| $modifiedLabel2 lda  #0     ; modified | ||||
|                 cmp  $varname | ||||
|                 bcc  $endLabel | ||||
|                 bcs  $loopLabel | ||||
| $endLabel""") | ||||
|         } else { | ||||
|             asmgen.out(""" | ||||
|                 lda  $varname | ||||
|                 clc | ||||
|                 adc  #<$stepsize | ||||
|                 sta  $varname | ||||
|                 lda  $varname+1 | ||||
|                 adc  #>$stepsize | ||||
|                 sta  $varname+1 | ||||
| $modifiedLabel2 lda  #0   ; modified | ||||
|                 cmp  $varname | ||||
| $modifiedLabel  lda  #0   ; modified | ||||
|                 sbc  $varname+1 | ||||
|                 bvc  + | ||||
|                 eor  #$80 | ||||
| +               bpl  $loopLabel                 | ||||
| $endLabel""") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun forOverWordsRangeStepGreaterOneDescending(range: PtRange, varname: String, iterableDt: DataType, loopLabel: String, endLabel: String, stmt: PtForLoop) { | ||||
|         // (u)words, step <= -2 | ||||
|         val stepsize = range.step.asConstInteger()!! | ||||
|         val modifiedLabel = asmgen.makeLabel("for_modified") | ||||
|         val modifiedLabel2 = asmgen.makeLabel("for_modifiedb") | ||||
|         asmgen.assignExpressionToRegister(range.to, RegisterOrPair.AY) | ||||
|         precheckFromToWord(iterableDt, stepsize, varname, endLabel) | ||||
|         asmgen.romableError("self-modifying code (forloop over words range)", stmt.position)  // TODO fix romable; there is self-modifying code below | ||||
|         asmgen.out(""" | ||||
|                 sty  $modifiedLabel+1 | ||||
|                 sta  $modifiedLabel2+1 | ||||
| $loopLabel""") | ||||
|         asmgen.translate(stmt.statements) | ||||
|         asmgen.out(""" | ||||
|                 lda  $varname | ||||
|                 sec | ||||
|                 sbc  #<${stepsize.absoluteValue} | ||||
|                 sta  $varname | ||||
|                 tax | ||||
|                 lda  $varname+1 | ||||
|                 sbc  #>${stepsize.absoluteValue} | ||||
|                 sta  $varname+1 | ||||
|                 txa | ||||
| $modifiedLabel2 cmp  #0    ; modified  | ||||
|                 lda  $varname+1 | ||||
| $modifiedLabel  sbc  #0    ; modified | ||||
|                 bvc  + | ||||
|                 eor  #$80 | ||||
| +               bpl  $loopLabel                 | ||||
| $endLabel""") | ||||
|     } | ||||
|  | ||||
|     private fun precheckFromToWord(iterableDt: DataType, stepsize: Int, fromVar: String, endLabel: String) { | ||||
|         // pre-check for end already reached. | ||||
|         // 'to' is in AY, do NOT clobber this! | ||||
|         if(iterableDt.isSignedWordArray) { | ||||
|             if(stepsize<0) | ||||
|                 asmgen.out(""" | ||||
|                     sta  P8ZP_SCRATCH_W2        ; to | ||||
|                     sty  P8ZP_SCRATCH_W2+1      ; to | ||||
|                     lda  $fromVar | ||||
|                     cmp  P8ZP_SCRATCH_W2 | ||||
|                     lda  $fromVar+1 | ||||
|                     sbc  P8ZP_SCRATCH_W2+1 | ||||
|                     bvc  + | ||||
|                     eor  #$80 | ||||
| +                   bmi  $endLabel | ||||
|                     lda  P8ZP_SCRATCH_W2 | ||||
|                     ldy  P8ZP_SCRATCH_W2+1""") | ||||
|             else | ||||
|                 asmgen.out(""" | ||||
|                     sta  P8ZP_SCRATCH_REG | ||||
|                     cmp  $fromVar | ||||
|                     tya | ||||
|                     sbc  $fromVar+1 | ||||
|                     bvc  + | ||||
|                     eor  #$80 | ||||
| +                   bmi  $endLabel | ||||
|                     lda  P8ZP_SCRATCH_REG""") | ||||
|         } else { | ||||
|             if(stepsize<0) | ||||
|                 asmgen.out(""" | ||||
|                     cpy  $fromVar+1 | ||||
|                     beq  + | ||||
|                     bcc  ++ | ||||
|                     bcs  $endLabel | ||||
| +                   cmp  $fromVar | ||||
|                     bcc  + | ||||
|                     beq  + | ||||
|                     bne  $endLabel | ||||
| +""") | ||||
|             else | ||||
|                 asmgen.out(""" | ||||
|                     cpy  $fromVar+1 | ||||
|                     bcc  $endLabel | ||||
|                     bne  + | ||||
|                     cmp  $fromVar | ||||
|                     bcc  $endLabel | ||||
| +""") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun translateForOverIterableVar(stmt: PtForLoop, iterableDt: DataType, ident: PtIdentifier) { | ||||
|         val loopLabel = asmgen.makeLabel("for_loop") | ||||
|         val endLabel = asmgen.makeLabel("for_end") | ||||
|         asmgen.loopEndLabels.add(endLabel) | ||||
|         val iterableName = asmgen.asmVariableName(ident) | ||||
|         val numElements: UInt = when(val symbol = asmgen.symbolTable.lookup(ident.name)) { | ||||
|             is StStaticVariable -> symbol.length!! | ||||
|             is StMemVar -> symbol.length!! | ||||
|             else -> 0u | ||||
|         } | ||||
|         when { | ||||
|             iterableDt.isString -> { | ||||
|                 if(asmgen.options.romable) { | ||||
|                     val indexVar = asmgen.createTempVarReused(BaseDataType.UBYTE, false, stmt) | ||||
|                     asmgen.out(""" | ||||
|                         ldy  #0 | ||||
|                         sty  $indexVar | ||||
| $loopLabel              lda  $iterableName,y | ||||
|                         beq  $endLabel | ||||
|                         sta  ${asmgen.asmVariableName(stmt.variable)}""") | ||||
|                     asmgen.translate(stmt.statements) | ||||
|                     asmgen.out(""" | ||||
|                         inc  $indexVar | ||||
|                         ldy  $indexVar | ||||
|                         bne  $loopLabel | ||||
| $endLabel""") | ||||
|                 } else { | ||||
|                     val indexVar = asmgen.makeLabel("for_index") | ||||
|                     asmgen.out(""" | ||||
|                         ldy  #0 | ||||
|                         sty  $indexVar | ||||
| $loopLabel              lda  $iterableName,y | ||||
|                         beq  $endLabel | ||||
|                         sta  ${asmgen.asmVariableName(stmt.variable)}""") | ||||
|                     asmgen.translate(stmt.statements) | ||||
|                     asmgen.out(""" | ||||
|                         inc  $indexVar | ||||
|                         ldy  $indexVar | ||||
|                         bne  $loopLabel | ||||
| $indexVar   .byte  0                         | ||||
| $endLabel""") | ||||
|                 } | ||||
|             } | ||||
|             iterableDt.isByteArray || iterableDt.isBoolArray -> { | ||||
|                 val indexVar = if(asmgen.options.romable) | ||||
|                     asmgen.createTempVarReused(iterableDt.elementType().base, false, stmt) | ||||
|                 else | ||||
|                     asmgen.makeLabel("for_index") | ||||
|                 asmgen.out(""" | ||||
|                     ldy  #0 | ||||
| $loopLabel          sty  $indexVar | ||||
|                     lda  $iterableName,y | ||||
|                     sta  ${asmgen.asmVariableName(stmt.variable)}""") | ||||
|                 asmgen.translate(stmt.statements) | ||||
|                 if(numElements<=255u) { | ||||
|                     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(!asmgen.options.romable) { | ||||
|                     if(numElements>=16u) { | ||||
|                         // allocate index var on ZP if possible, otherwise inline | ||||
|                         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) | ||||
|             } | ||||
|             iterableDt.isSplitWordArray -> { | ||||
|                 val indexVar = if(asmgen.options.romable) | ||||
|                     asmgen.createTempVarReused(BaseDataType.UBYTE, false, stmt) | ||||
|                 else | ||||
|                     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<=255u) { | ||||
|                     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(!asmgen.options.romable) { | ||||
|                     if(numElements>=16u) { | ||||
|                         // allocate index var on ZP if possible, otherwise inline | ||||
|                         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) | ||||
|             } | ||||
|             iterableDt.isWordArray -> { | ||||
|                 val indexVar = if(asmgen.options.romable) | ||||
|                     asmgen.createTempVarReused(BaseDataType.UBYTE, false, stmt) | ||||
|                 else | ||||
|                     asmgen.makeLabel("for_index") | ||||
|                 val loopvarName = asmgen.asmVariableName(stmt.variable) | ||||
|                 asmgen.out(""" | ||||
|                     ldy  #0 | ||||
| $loopLabel          sty  $indexVar | ||||
|                     lda  $iterableName,y | ||||
|                     sta  $loopvarName | ||||
|                     lda  $iterableName+1,y | ||||
|                     sta  $loopvarName+1""") | ||||
|                 asmgen.translate(stmt.statements) | ||||
|                 if(numElements<=127u) { | ||||
|                     asmgen.out(""" | ||||
|                         ldy  $indexVar | ||||
|                         iny | ||||
|                         iny | ||||
|                         cpy  #${numElements*2u} | ||||
|                         beq  $endLabel | ||||
|                         bne  $loopLabel""") | ||||
|                 } else { | ||||
|                     // array size is 128 words, 256 bytes | ||||
|                     asmgen.out(""" | ||||
|                         ldy  $indexVar | ||||
|                         iny | ||||
|                         iny | ||||
|                         bne  $loopLabel | ||||
|                         beq  $endLabel""") | ||||
|                 } | ||||
|                 if(!asmgen.options.romable) { | ||||
|                     if(numElements>=16u) { | ||||
|                         // allocate index var on ZP if possible, otherwise inline | ||||
|                         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) | ||||
|             } | ||||
|             iterableDt.isFloatArray -> { | ||||
|                 throw AssemblyError("for loop with floating point variables is not supported") | ||||
|             } | ||||
|             else -> throw AssemblyError("can't iterate over $iterableDt") | ||||
|         } | ||||
|         asmgen.loopEndLabels.removeLast() | ||||
|     } | ||||
|  | ||||
|     private fun translateForOverConstRange(stmt: PtForLoop, iterableDt: DataType, range: IntProgression) { | ||||
|         if (range.isEmpty() || range.step==0) | ||||
|             throw AssemblyError("empty range or step 0") | ||||
|         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 translateForSimpleByteRangeDesc(stmt, range, iterableDt.isUnsignedByteArray) | ||||
|         } | ||||
|         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 translateForSimpleWordRangeDesc(stmt, range) | ||||
|         } | ||||
|  | ||||
|         // not one of the easy cases, generate more complex code... | ||||
|         val loopLabel = asmgen.makeLabel("for_loop") | ||||
|         val endLabel = asmgen.makeLabel("for_end") | ||||
|         asmgen.loopEndLabels.add(endLabel) | ||||
|         when { | ||||
|             iterableDt.isByteArray -> { | ||||
|                 // loop over byte range via loopvar, step >= 2 or <= -2 | ||||
|                 val varname = asmgen.asmVariableName(stmt.variable) | ||||
|                 asmgen.out(""" | ||||
|                             lda  #${range.first} | ||||
|                             sta  $varname | ||||
| $loopLabel""") | ||||
|                 asmgen.translate(stmt.statements) | ||||
|                 when (range.step) { | ||||
|                     0, 1, -1 -> { | ||||
|                         throw AssemblyError("step 0, 1 and -1 should have been handled specifically  $range ${stmt.position}") | ||||
|                     } | ||||
|                     2 -> { | ||||
|                         if(range.last==255 || range.last==254) { | ||||
|                             asmgen.out(""" | ||||
|                                 inc  $varname | ||||
|                                 beq  $endLabel | ||||
|                                 inc  $varname | ||||
|                                 bne  $loopLabel""") | ||||
|                         } else { | ||||
|                             asmgen.out(""" | ||||
|                                 inc  $varname | ||||
|                                 inc  $varname | ||||
|                                 lda  $varname | ||||
|                                 cmp  #${range.last+2} | ||||
|                                 bne  $loopLabel""") | ||||
|                         } | ||||
|                     } | ||||
|                     -2 -> { | ||||
|                         when (range.last) { | ||||
|                             0 -> { | ||||
|                                 asmgen.out(""" | ||||
|                                     lda  $varname | ||||
|                                     beq  $endLabel | ||||
|                                     dec  $varname | ||||
|                                     dec  $varname""") | ||||
|                                 asmgen.jmp(loopLabel) | ||||
|                             } | ||||
|                             1 -> asmgen.out(""" | ||||
|                                     dec  $varname | ||||
|                                     beq  $endLabel | ||||
|                                     dec  $varname | ||||
|                                     bne  $loopLabel""") | ||||
|                             else -> asmgen.out(""" | ||||
|                                     dec  $varname | ||||
|                                     dec  $varname | ||||
|                                     lda  $varname | ||||
|                                     cmp  #${range.last-2} | ||||
|                                     bne  $loopLabel""") | ||||
|                         } | ||||
|                     } | ||||
|                     else -> { | ||||
|                         // step <= -3 or >= 3 | ||||
|                         asmgen.out(""" | ||||
|                             lda  $varname | ||||
|                             cmp  #${range.last} | ||||
|                             beq  $endLabel | ||||
|                             clc | ||||
|                             adc  #${range.step} | ||||
|                             sta  $varname""") | ||||
|                         asmgen.jmp(loopLabel) | ||||
|                     } | ||||
|                 } | ||||
|                 asmgen.out(endLabel) | ||||
|             } | ||||
|             iterableDt.isWordArray && !iterableDt.isSplitWordArray -> { | ||||
|                 // loop over word range via loopvar, step >= 2 or <= -2 | ||||
|                 val varname = asmgen.asmVariableName(stmt.variable) | ||||
|                 when (range.step) { | ||||
|                     0, 1, -1 -> { | ||||
|                         throw AssemblyError("step 0, 1 and -1 should have been handled specifically  $stmt") | ||||
|                     } | ||||
|                     else -> { | ||||
|                         // word, step >= 2 or <= -2 | ||||
|                         // note: range.last has already been adjusted by kotlin itself to actually be the last value of the sequence | ||||
|                         asmgen.out(""" | ||||
|                             lda  #<${range.first} | ||||
|                             ldy  #>${range.first} | ||||
|                             sta  $varname | ||||
|                             sty  $varname+1 | ||||
| $loopLabel""") | ||||
|                         asmgen.translate(stmt.statements) | ||||
|                         asmgen.out(""" | ||||
|                             lda  $varname | ||||
|                             cmp  #<${range.last} | ||||
|                             bne  + | ||||
|                             lda  $varname+1 | ||||
|                             cmp  #>${range.last} | ||||
|                             bne  + | ||||
|                             beq  $endLabel | ||||
| +                           lda  $varname | ||||
|                             clc | ||||
|                             adc  #<${range.step} | ||||
|                             sta  $varname | ||||
|                             lda  $varname+1 | ||||
|                             adc  #>${range.step} | ||||
|                             sta  $varname+1""") | ||||
|                         asmgen.jmp(loopLabel) | ||||
|                         asmgen.out(endLabel) | ||||
|                     } | ||||
|                 } | ||||
|             } | ||||
|             else -> throw AssemblyError("range expression can only be byte or word") | ||||
|         } | ||||
|         asmgen.loopEndLabels.removeLast() | ||||
|     } | ||||
|  | ||||
|     private fun translateForSimpleByteRangeAsc(stmt: PtForLoop, range: IntProgression) { | ||||
|         val loopLabel = asmgen.makeLabel("for_loop") | ||||
|         val endLabel = asmgen.makeLabel("for_end") | ||||
|         asmgen.loopEndLabels.add(endLabel) | ||||
|         val varname = asmgen.asmVariableName(stmt.variable) | ||||
|         asmgen.out(""" | ||||
|                 lda  #${range.first} | ||||
|                 sta  $varname | ||||
| $loopLabel""") | ||||
|         asmgen.translate(stmt.statements) | ||||
|         if (range.last == 255) { | ||||
|             asmgen.out(""" | ||||
|                 inc  $varname | ||||
|                 bne  $loopLabel | ||||
| $endLabel""") | ||||
|         } else { | ||||
|             asmgen.out(""" | ||||
|                 inc  $varname | ||||
|                 lda  $varname | ||||
|                 cmp  #${range.last+1} | ||||
|                 bne  $loopLabel | ||||
| $endLabel""") | ||||
|         } | ||||
|         asmgen.loopEndLabels.removeLast() | ||||
|     } | ||||
|  | ||||
|     private fun translateForSimpleByteRangeDesc(stmt: PtForLoop, range: IntProgression, unsigned: Boolean) { | ||||
|         val loopLabel = asmgen.makeLabel("for_loop") | ||||
|         val varname = asmgen.asmVariableName(stmt.variable) | ||||
|         asmgen.out(""" | ||||
|             lda  #${range.first} | ||||
|             sta  $varname | ||||
| $loopLabel""") | ||||
|         asmgen.translate(stmt.statements) | ||||
|         when (range.last) { | ||||
|             0 -> { | ||||
|                 if(!unsigned || range.first<=127) { | ||||
|                     asmgen.out(""" | ||||
|                         dec  $varname | ||||
|                         bpl  $loopLabel""") | ||||
|                 } else { | ||||
|                     asmgen.out(""" | ||||
|                         dec  $varname | ||||
|                         lda  $varname | ||||
|                         cmp  #255 | ||||
|                         bne  $loopLabel""") | ||||
|                 } | ||||
|             } | ||||
|             1 -> { | ||||
|                 asmgen.out(""" | ||||
|                     dec  $varname | ||||
|                     bne  $loopLabel""") | ||||
|             } | ||||
|             else -> { | ||||
|                 asmgen.out(""" | ||||
|                     dec  $varname | ||||
|                     lda  $varname | ||||
|                     cmp  #${range.last-1} | ||||
|                     bne  $loopLabel""") | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun translateForSimpleWordRangeAsc(stmt: PtForLoop, range: IntProgression) { | ||||
|         val loopLabel = asmgen.makeLabel("for_loop") | ||||
|         val endLabel = asmgen.makeLabel("for_end") | ||||
|         asmgen.loopEndLabels.add(endLabel) | ||||
|         val varname = asmgen.asmVariableName(stmt.variable) | ||||
|         asmgen.out(""" | ||||
|             lda  #<${range.first} | ||||
|             ldy  #>${range.first} | ||||
|             sta  $varname | ||||
|             sty  $varname+1 | ||||
| $loopLabel""") | ||||
|         asmgen.translate(stmt.statements) | ||||
|         asmgen.out(""" | ||||
|             lda  $varname | ||||
|             cmp  #<${range.last} | ||||
|             bne  + | ||||
|             lda  $varname+1 | ||||
|             cmp  #>${range.last} | ||||
|             beq  $endLabel | ||||
| +           inc  $varname | ||||
|             bne  $loopLabel | ||||
|             inc  $varname+1""") | ||||
|         asmgen.jmp(loopLabel) | ||||
|         asmgen.out(endLabel) | ||||
|         asmgen.loopEndLabels.removeLast() | ||||
|     } | ||||
|  | ||||
|     private fun translateForSimpleWordRangeDesc(stmt: PtForLoop, range: IntProgression) { | ||||
|         val loopLabel = asmgen.makeLabel("for_loop") | ||||
|         val endLabel = asmgen.makeLabel("for_end") | ||||
|         asmgen.loopEndLabels.add(endLabel) | ||||
|         val varname = asmgen.asmVariableName(stmt.variable) | ||||
|         asmgen.out(""" | ||||
|             lda  #<${range.first} | ||||
|             ldy  #>${range.first} | ||||
|             sta  $varname | ||||
|             sty  $varname+1 | ||||
| $loopLabel""") | ||||
|         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(""" | ||||
| +           lda  $varname | ||||
|             bne  + | ||||
|             dec  $varname+1 | ||||
| +           dec  $varname""") | ||||
|         asmgen.jmp(loopLabel) | ||||
|         asmgen.out(endLabel) | ||||
|         asmgen.loopEndLabels.removeLast() | ||||
|     } | ||||
|  | ||||
|     private fun assignLoopvarWord(stmt: PtForLoop, range: PtRange) = | ||||
|         asmgen.assignExpressionToVariable( | ||||
|             range.from, | ||||
|             asmgen.asmVariableName(stmt.variable), | ||||
|             stmt.variable.type) | ||||
| } | ||||
							
								
								
									
										380
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/FunctionCallAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										380
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/FunctionCallAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,380 @@ | ||||
| package prog8.codegen.cpu6502 | ||||
|  | ||||
| import prog8.code.StNodeType | ||||
| import prog8.code.ast.* | ||||
| import prog8.code.core.* | ||||
| import prog8.codegen.cpu6502.assignment.AsmAssignSource | ||||
| import prog8.codegen.cpu6502.assignment.AsmAssignTarget | ||||
| import prog8.codegen.cpu6502.assignment.AsmAssignment | ||||
| import prog8.codegen.cpu6502.assignment.TargetStorageKind | ||||
|  | ||||
|  | ||||
| internal class FunctionCallAsmGen(private val program: PtProgram, private val asmgen: AsmGen6502Internal) { | ||||
|  | ||||
|     internal fun translateFunctionCallStatement(stmt: PtFunctionCall) { | ||||
|         translateFunctionCall(stmt) | ||||
|         // just ignore any result values from the function call. | ||||
|     } | ||||
|  | ||||
|     internal fun optimizeIntArgsViaCpuRegisters(params: List<PtSubroutineParameter>): Boolean { | ||||
|         // When the parameter(s) are passed via an explicit register or register pair, | ||||
|         // we consider them NOT to be optimized into (possibly different) CPU registers. | ||||
|         // Just load them in whatever the register spec says. | ||||
|         return when (params.size) { | ||||
|             1 -> params[0].register == null && (params[0].type.isIntegerOrBool || params[0].type.isPointer) | ||||
|             2 -> params[0].type.isByteOrBool && params[1].type.isByteOrBool && params[0].register == null && params[1].register == null | ||||
|             else -> false | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     internal fun translateFunctionCall(call: PtFunctionCall) { | ||||
|         // Output only the code to set up the parameters and perform the actual call | ||||
|         // NOTE: does NOT output the code to deal with the result values! | ||||
|         // NOTE: does NOT output code to save/restore the X register for this call! Every caller should deal with this in their own way!! | ||||
|         //       (you can use subroutine.shouldSaveX() and saveX()/restoreX() routines as a help for this) | ||||
|  | ||||
|         val symbol = asmgen.symbolTable.lookup(call.name)!! | ||||
|         if(symbol.type == StNodeType.LABEL) { | ||||
|             require(call.void) | ||||
|             asmgen.out("  jsr  ${asmgen.asmSymbolName(symbol.scopedNameString)}") | ||||
|             return | ||||
|         } | ||||
|  | ||||
|         val sub = symbol.astNode as IPtSubroutine | ||||
|         val subAsmName = asmgen.asmSymbolName(call.name) | ||||
|  | ||||
|         if(sub is PtAsmSub) { | ||||
|             argumentsViaRegisters(sub, call) | ||||
|             if (sub.inline) { | ||||
|                 // inline the subroutine. (regardless of optimization settings!) | ||||
|                 // we do this by copying the subroutine's statements at the call site. | ||||
|                 // NOTE: *if* there is a return statement, it will be the only one, and the very last statement of the subroutine | ||||
|                 // (this condition has been enforced by an ast check earlier) | ||||
|                 asmgen.out("  \t; inlined routine follows: ${sub.name}") | ||||
|                 sub.children.forEach { asmgen.translate(it as PtInlineAssembly) } | ||||
|                 asmgen.out("  \t; inlined routine end: ${sub.name}") | ||||
|             } else { | ||||
|                 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) { | ||||
|                         if(asmgen.options.romable) | ||||
|                             TODO("no codegen yet for non-const bank in subroutine call that's usable in ROM  ${call.position}") | ||||
|  | ||||
|                         // self-modifying code: set jsrfar bank argument | ||||
|                         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) { | ||||
|             val parameters = sub.signature.children.filterIsInstance<PtSubroutineParameter>() | ||||
|             if(optimizeIntArgsViaCpuRegisters(parameters)) { | ||||
|                 // Note that if the args fit into cpu registers, we don't concern ourselves here | ||||
|                 // if they should be put into regular subroutine parameter variables, or the R0-R15 register variables. | ||||
|                 // That is now up to the subroutine itself. | ||||
|                 useCpuRegistersForArgs(call.args, sub) | ||||
|             } else { | ||||
|                 // arguments via variables | ||||
|                 val paramValues = parameters.zip(call.args) | ||||
|                 val (normalParams, registerParams) = paramValues.partition { (it.first.register == null) } | ||||
|                 if (normalParams.isNotEmpty()) { | ||||
|                     for (p in normalParams) | ||||
|                         argumentViaVariable(sub, p.first, p.second) | ||||
|                 } | ||||
|                 if (registerParams.isNotEmpty()) { | ||||
|                     // the R0-R15 'registers' are not really registers. They're just special variables. | ||||
|                     for (p in registerParams) | ||||
|                         argumentViaVariable(sub, p.first, p.second) | ||||
|                 } | ||||
|             } | ||||
|             asmgen.out("  jsr  $subAsmName") | ||||
|         } | ||||
|         else throw AssemblyError("invalid sub type") | ||||
|  | ||||
|         // 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.signature.children.filterIsInstance<PtSubroutineParameter>() | ||||
|         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 | ||||
|                     if(asmgen.needAsaveForExpr(args[0]) && !asmgen.needAsaveForExpr(args[1])) { | ||||
|                         // first 0 then 1 | ||||
|                         argumentViaRegister(sub, IndexedValue(0, params[0]), args[0], RegisterOrPair.A) | ||||
|                         argumentViaRegister(sub, IndexedValue(1, params[1]), args[1], RegisterOrPair.Y) | ||||
|                     } else if(!asmgen.needAsaveForExpr(args[0]) && asmgen.needAsaveForExpr(args[1])) { | ||||
|                         // first 1 then 0 | ||||
|                         argumentViaRegister(sub, IndexedValue(1, params[1]), args[1], RegisterOrPair.Y) | ||||
|                         argumentViaRegister(sub, IndexedValue(0, params[0]), args[0], RegisterOrPair.A) | ||||
|                     } else { | ||||
|                         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 { | ||||
|         return when(arg) { | ||||
|             is PtBuiltinFunctionCall -> { | ||||
|                 if (arg.name in arrayOf("lsb", "msb", "lsw", "msw")) | ||||
|                     return usesOtherRegistersWhileEvaluating(arg.args[0]) | ||||
|                 if (arg.name == "mkword") | ||||
|                     return usesOtherRegistersWhileEvaluating(arg.args[0]) || usesOtherRegistersWhileEvaluating(arg.args[1]) | ||||
|                 return !arg.isSimple() | ||||
|             } | ||||
|             is PtAddressOf -> false | ||||
|             is PtIdentifier -> false | ||||
|             is PtIrRegister -> false | ||||
|             is PtMemoryByte -> arg.address !is PtNumber && arg.address !is PtIdentifier | ||||
|             is PtNumber -> false | ||||
|             is PtBool -> false | ||||
|             else -> true | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun argumentsViaRegisters(sub: PtAsmSub, call: PtFunctionCall) { | ||||
|         val registersUsed = mutableListOf<RegisterOrStatusflag>() | ||||
|  | ||||
|         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 usedY() = registersUsed.any {it.registerOrPair==RegisterOrPair.Y || it.registerOrPair==RegisterOrPair.AY || it.registerOrPair==RegisterOrPair.XY} | ||||
|  | ||||
|         if(sub.parameters.size==1) { | ||||
|             argumentViaRegister(sub, IndexedValue(0, sub.parameters.single().second), call.args[0]) | ||||
|         } else { | ||||
|             val optimalEvalOrder = asmsub6502ArgsEvalOrder(sub) | ||||
|             optimalEvalOrder.forEach { | ||||
|                 val param = sub.parameters[it] | ||||
|                 val arg = call.args[it] | ||||
|                 registersUsed += if(usesOtherRegistersWhileEvaluating(arg)) { | ||||
|                     if(!registersUsed.any{r -> r.statusflag!=null || r.registerOrPair in CpuRegisters}) | ||||
|                         argumentViaRegister(sub, IndexedValue(it, param.second), arg) | ||||
|                     else if(registersUsed.any { r-> r.statusflag!=null }) { | ||||
|                         throw AssemblyError("call argument evaluation problem: can't save cpu statusregister parameter ${call.position}") | ||||
|                     } | ||||
|                     else { | ||||
|                         if(usedX()) asmgen.saveRegisterStack(CpuRegister.X, usedA()) | ||||
|                         if(usedY()) asmgen.saveRegisterStack(CpuRegister.Y, usedA()) | ||||
|                         if(usedA()) asmgen.saveRegisterStack(CpuRegister.A, usedA()) | ||||
|                         val used = argumentViaRegister(sub, IndexedValue(it, param.second), arg) | ||||
|                         if(usedA()) asmgen.restoreRegisterStack(CpuRegister.A, false) | ||||
|                         if(usedY()) asmgen.restoreRegisterStack(CpuRegister.Y, true) | ||||
|                         if(usedX()) asmgen.restoreRegisterStack(CpuRegister.X, true) | ||||
|                         used | ||||
|                     } | ||||
|                 } else { | ||||
|                     argumentViaRegister(sub, IndexedValue(it, param.second), arg) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun argumentViaVariable(sub: PtSub, parameter: PtSubroutineParameter, value: PtExpression) { | ||||
|         // pass parameter via a regular variable (not via registers) | ||||
|         if(!isArgumentTypeCompatible(value.type, parameter.type)) | ||||
|             throw AssemblyError("argument type incompatible") | ||||
|  | ||||
|         val reg = parameter.register | ||||
|         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 { | ||||
|         // pass argument via a register parameter | ||||
|         if(!isArgumentTypeCompatible(value.type, parameter.value.type)) | ||||
|             throw AssemblyError("argument type incompatible") | ||||
|  | ||||
|         val paramRegister: RegisterOrStatusflag = when(sub) { | ||||
|             is PtAsmSub -> if(registerOverride==null) sub.parameters[parameter.index].first else RegisterOrStatusflag(registerOverride, null) | ||||
|             is PtSub -> RegisterOrStatusflag(registerOverride!!, null) | ||||
|         } | ||||
|         val statusflag = paramRegister.statusflag | ||||
|         val register = paramRegister.registerOrPair | ||||
|         val requiredDt = parameter.value.type | ||||
|         if(requiredDt!=value.type) { | ||||
|             if(value.type.largerSizeThan(requiredDt)) | ||||
|                 throw AssemblyError("can only convert byte values to word param types") | ||||
|         } | ||||
|         if (statusflag!=null) { | ||||
|             if(requiredDt!=value.type) | ||||
|                 throw AssemblyError("for statusflag, byte or bool value is required") | ||||
|             if (statusflag == Statusflag.Pc) { | ||||
|                 // this boolean param needs to be set last, right before the jsr | ||||
|                 // for now, this is already enforced on the subroutine definition by the Ast Checker | ||||
|                 when(value) { | ||||
|                     is PtNumber -> { | ||||
|                         val carrySet = value.number.toInt() != 0 | ||||
|                         asmgen.out(if(carrySet) "  sec" else "  clc") | ||||
|                     } | ||||
|                     is PtBool -> { | ||||
|                         asmgen.out(if(value.value) "  sec" else "  clc") | ||||
|                     } | ||||
|                     is PtIdentifier -> { | ||||
|                         val sourceName = asmgen.asmVariableName(value) | ||||
|                         // note: cannot use X register here to store A because it might be used for other arguments | ||||
|                         asmgen.out(""" | ||||
|                             pha | ||||
|                             clc | ||||
|                             lda  $sourceName | ||||
|                             beq  + | ||||
|                             sec   | ||||
| +                           pla""") | ||||
|                     } | ||||
|                     else -> { | ||||
|                         asmgen.assignExpressionToRegister(value, RegisterOrPair.A) | ||||
|                         asmgen.out("  ror  a") | ||||
|                     } | ||||
|                 } | ||||
|             } else throw AssemblyError("can only use Carry as status flag parameter") | ||||
|             return RegisterOrStatusflag(null, statusflag) | ||||
|         } | ||||
|         else { | ||||
|             // via register or register pair | ||||
|             register!! | ||||
|             if(requiredDt.largerSizeThan(value.type)) { | ||||
|                 // we need to sign extend the source, do this via temporary word variable | ||||
|                 asmgen.assignExpressionToVariable(value, "P8ZP_SCRATCH_W1", DataType.UBYTE) | ||||
|                 asmgen.signExtendVariableLsb("P8ZP_SCRATCH_W1", value.type.base) | ||||
|                 asmgen.assignVariableToRegister("P8ZP_SCRATCH_W1", register, null, Position.DUMMY) | ||||
|             } else { | ||||
|                 val scope = value.definingISub() | ||||
|                 val target: AsmAssignTarget = | ||||
|                     if(parameter.value.type.isByte && register.isWord()) | ||||
|                         AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, parameter.value.type, scope, value.position, register = register) | ||||
|                     else { | ||||
|                         AsmAssignTarget.fromRegisters(register, parameter.value.type.isSigned, value.position, scope, asmgen) | ||||
|                     } | ||||
|                 val src = if(value.type.isPassByRef) { | ||||
|                     if(value is PtIdentifier) { | ||||
|                         val addr = PtAddressOf(value.type.typeForAddressOf(false), false, Position.DUMMY) | ||||
|                         addr.add(value) | ||||
|                         addr.parent = scope as PtNode | ||||
|                         AsmAssignSource.fromAstSource(addr, program, asmgen).adjustSignedUnsigned(target) | ||||
|                     } else { | ||||
|                         AsmAssignSource.fromAstSource(value, program, asmgen).adjustSignedUnsigned(target) | ||||
|                     } | ||||
|                 } else { | ||||
|                     AsmAssignSource.fromAstSource(value, program, asmgen).adjustSignedUnsigned(target) | ||||
|                 } | ||||
|                 asmgen.translateNormalAssignment(AsmAssignment(src, listOf(target), program.memsizer, target.position), scope) | ||||
|             } | ||||
|             return RegisterOrStatusflag(register, null) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun isArgumentTypeCompatible(argType: DataType, paramType: DataType): Boolean { | ||||
|         if(argType isAssignableTo paramType) | ||||
|             return true | ||||
|         if(argType.isBool && paramType.isBool) | ||||
|             return true | ||||
|         if(argType.isByte && paramType.isByte) | ||||
|             return true | ||||
|         if(argType.isWord && paramType.isWord) | ||||
|             return true | ||||
|  | ||||
|         // we have a special rule for some types. | ||||
|         // strings are assignable to UWORD, for example, and vice versa | ||||
|         if(argType.isString && paramType.isUnsignedWord) | ||||
|             return true | ||||
|         if(argType.isUnsignedWord && paramType.isString) | ||||
|             return true | ||||
|  | ||||
|         return false | ||||
|     } | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										1892
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/IfElseAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1892
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/IfElseAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										427
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/IfExpressionAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										427
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/IfExpressionAsmGen.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,427 @@ | ||||
| package prog8.codegen.cpu6502 | ||||
|  | ||||
| import prog8.code.ast.* | ||||
| import prog8.code.core.* | ||||
| 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 || | ||||
|                 target.datatype.isUnsignedWord && (expr.type.isString || expr.type.isPointer) || | ||||
|                 target.datatype.isPointer && (expr.type.isUnsignedWord || expr.type.isPointer || expr.type.isString)) | ||||
|         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) | ||||
|                 asmgen.jmp(endLabel) | ||||
|                 asmgen.out(falseLabel) | ||||
|                 asmgen.assignExpressionToRegister(expr.falsevalue, RegisterOrPair.A) | ||||
|                 asmgen.out(endLabel) | ||||
|                 assignmentAsmGen.assignRegisterByte(target, CpuRegister.A, false, false) | ||||
|             } | ||||
|             expr.type.isWord || expr.type.isString -> { | ||||
|                 asmgen.assignExpressionToRegister(expr.truevalue, RegisterOrPair.AY) | ||||
|                 asmgen.jmp(endLabel) | ||||
|                 asmgen.out(falseLabel) | ||||
|                 asmgen.assignExpressionToRegister(expr.falsevalue, RegisterOrPair.AY) | ||||
|                 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") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     internal fun assignBranchCondExpression(target: AsmAssignTarget, expr: PtBranchCondExpression) { | ||||
|         require(target.datatype==expr.type || | ||||
|                 target.datatype.isUnsignedWord && (expr.type.isString || expr.type.isPointer) || | ||||
|                 target.datatype.isPointer && (expr.type.isUnsignedWord || expr.type.isPointer || expr.type.isString)) | ||||
|  | ||||
|         if(target.kind==TargetStorageKind.REGISTER && target.datatype.isUnsignedByte && target.register==RegisterOrPair.A) { | ||||
|             if(expr.condition==BranchCondition.CC) { | ||||
|                 if(expr.truevalue.asConstInteger()==0 && expr.falsevalue.asConstInteger()==1) { | ||||
|                     asmgen.out("  lda  #0 |  rol a") | ||||
|                     return | ||||
|                 } | ||||
|                 else if(expr.truevalue.asConstInteger()==1 && expr.falsevalue.asConstInteger()==0) { | ||||
|                     asmgen.out("  lda  #0 |  rol a |  eor  #1") | ||||
|                     return | ||||
|                 } | ||||
|             } | ||||
|             else if(expr.condition==BranchCondition.CS) { | ||||
|                 if(expr.truevalue.asConstInteger()==0 && expr.falsevalue.asConstInteger()==1) { | ||||
|                     asmgen.out("  lda  #0 |  rol a |  eor  #1") | ||||
|                     return | ||||
|                 } | ||||
|                 else if(expr.truevalue.asConstInteger()==1 && expr.falsevalue.asConstInteger()==0) { | ||||
|                     asmgen.out("  lda  #0 |  rol a") | ||||
|                     return | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         val trueLabel = asmgen.makeLabel("branchexpr_true") | ||||
|         val endLabel = asmgen.makeLabel("branchexpr_end") | ||||
|         val branch = asmgen.branchInstruction(expr.condition, false) | ||||
|  | ||||
|         asmgen.out("  $branch  $trueLabel") | ||||
|  | ||||
|         when { | ||||
|             expr.type.isByteOrBool -> { | ||||
|                 asmgen.assignExpressionToRegister(expr.falsevalue, RegisterOrPair.A) | ||||
|                 asmgen.jmp(endLabel) | ||||
|                 asmgen.out(trueLabel) | ||||
|                 asmgen.assignExpressionToRegister(expr.truevalue, RegisterOrPair.A) | ||||
|                 asmgen.out(endLabel) | ||||
|                 assignmentAsmGen.assignRegisterByte(target, CpuRegister.A, false, false) | ||||
|             } | ||||
|             expr.type.isWord || expr.type.isString -> { | ||||
|                 asmgen.assignExpressionToRegister(expr.falsevalue, RegisterOrPair.AY) | ||||
|                 asmgen.jmp(endLabel) | ||||
|                 asmgen.out(trueLabel) | ||||
|                 asmgen.assignExpressionToRegister(expr.truevalue, RegisterOrPair.AY) | ||||
|                 asmgen.out(endLabel) | ||||
|                 assignmentAsmGen.assignRegisterpairWord(target, RegisterOrPair.AY) | ||||
|             } | ||||
|             expr.type.isFloat -> { | ||||
|                 asmgen.assignExpressionToRegister(expr.falsevalue, RegisterOrPair.FAC1, true) | ||||
|                 asmgen.jmp(endLabel) | ||||
|                 asmgen.out(trueLabel) | ||||
|                 asmgen.assignExpressionToRegister(expr.truevalue, RegisterOrPair.FAC1, true) | ||||
|                 asmgen.out(endLabel) | ||||
|                 asmgen.assignRegister(RegisterOrPair.FAC1, target) | ||||
|             } | ||||
|             else -> throw AssemblyError("weird dt") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun evalIfExpressionConditonAndBranchWhenFalse(condition: PtExpression, falseLabel: String) { | ||||
|         when (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") | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             is PtPrefix if condition.operator=="not" -> { | ||||
|                 throw AssemblyError("not prefix in ifexpression should have been replaced by swapped values") | ||||
|             } | ||||
|  | ||||
|             else -> { | ||||
|                 // 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.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) | ||||
|             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) | ||||
|             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) | ||||
|             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) | ||||
|             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) | ||||
|             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) | ||||
|             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 | ||||
|  | ||||
|         val useBIT = asmgen.checkIfConditionCanUseBIT(condition) | ||||
|         if(useBIT!=null) { | ||||
|             // use a BIT instruction to test for bit 7 or 6 set/clear | ||||
|             val (testForBitSet, variable, bitmask) = useBIT | ||||
|             when (bitmask) { | ||||
|                 128 -> { | ||||
|                     // test via bit + N flag | ||||
|                     asmgen.out("  bit  ${variable.name}") | ||||
|                     if(testForBitSet) asmgen.out("  bpl  $falseLabel") | ||||
|                     else asmgen.out("  bmi  $falseLabel") | ||||
|                     return | ||||
|                 } | ||||
|                 64 -> { | ||||
|                     // 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("BIT can only work on bits 7 and 6") | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         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") | ||||
|         } | ||||
|     } | ||||
|  | ||||
| } | ||||
							
								
								
									
										1042
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/ProgramAndVarsGen.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1042
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/ProgramAndVarsGen.kt
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										132
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/VariableAllocator.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										132
									
								
								codeGenCpu6502/src/prog8/codegen/cpu6502/VariableAllocator.kt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,132 @@ | ||||
| package prog8.codegen.cpu6502 | ||||
|  | ||||
| import com.github.michaelbull.result.fold | ||||
| import com.github.michaelbull.result.onSuccess | ||||
| import prog8.code.* | ||||
| import prog8.code.core.* | ||||
|  | ||||
|  | ||||
| internal class VariableAllocator(private val symboltable: SymbolTable, | ||||
|                                  private val options: CompilationOptions, | ||||
|                                  private val errors: IErrorReporter | ||||
| ) { | ||||
|  | ||||
|     private val zeropage = options.compTarget.zeropage | ||||
|     internal val globalFloatConsts = mutableMapOf<Double, String>()     // all float values in the entire program (value -> varname) | ||||
|     internal val zeropageVars: Map<String, MemoryAllocator.VarAllocation> | ||||
|  | ||||
|     init { | ||||
|         allocateZeropageVariables() | ||||
|         zeropageVars = zeropage.allocatedVariables | ||||
|     } | ||||
|  | ||||
|     internal fun isZpVar(scopedName: String): Boolean { | ||||
|         if(scopedName in zeropageVars) | ||||
|             return true | ||||
|  | ||||
|         val v = symboltable.lookup(scopedName) | ||||
|         return if(v is StMemVar) v.address <= 255u else false | ||||
|     } | ||||
|  | ||||
|     internal fun getFloatAsmConst(number: Double): String { | ||||
|         val asmName = globalFloatConsts[number] | ||||
|         if(asmName!=null) | ||||
|             return asmName | ||||
|  | ||||
|         val newName = "prog8_float_const_${globalFloatConsts.size}" | ||||
|         globalFloatConsts[number] = newName | ||||
|         return newName | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Allocate variables into the Zeropage. | ||||
|      * The result should be retrieved from the current machine's zeropage object! | ||||
|      */ | ||||
|     private fun allocateZeropageVariables() { | ||||
|         if(options.zeropage== ZeropageType.DONTUSE) | ||||
|             return | ||||
|  | ||||
|         val allVariables = collectAllVariables(symboltable) | ||||
|  | ||||
|         val numberOfAllocatableVariables = allVariables.size | ||||
|         val varsRequiringZp = allVariables.filter { it.zpwish == ZeropageWish.REQUIRE_ZEROPAGE } | ||||
|         val varsPreferringZp = allVariables.filter { it.zpwish == ZeropageWish.PREFER_ZEROPAGE } | ||||
|         val varsNotZp = allVariables.filter { it.zpwish == ZeropageWish.NOT_IN_ZEROPAGE } | ||||
|         val (varsDontCareWithoutAlignment, varsDontCareWithAlignment) = allVariables.filter { it.zpwish == ZeropageWish.DONTCARE }.partition { it.align == 0u } | ||||
|         require(varsDontCareWithAlignment.size + varsDontCareWithoutAlignment.size + varsRequiringZp.size + varsPreferringZp.size + varsNotZp.size == numberOfAllocatableVariables) | ||||
|  | ||||
|         var numVariablesAllocatedInZP = 0 | ||||
|         var numberOfNonIntegerVariables = 0 | ||||
|  | ||||
|         varsRequiringZp.forEach { variable -> | ||||
|             val result = zeropage.allocate( | ||||
|                 variable.scopedNameString, | ||||
|                 variable.dt, | ||||
|                 variable.length?.toInt(), | ||||
|                 variable.astNode?.position ?: Position.DUMMY, | ||||
|                 errors | ||||
|             ) | ||||
|             result.fold( | ||||
|                 success = { | ||||
|                     numVariablesAllocatedInZP++ | ||||
|                 }, | ||||
|                 failure = { | ||||
|                     errors.err(it.message!!, variable.astNode?.position ?: Position.DUMMY) | ||||
|                 } | ||||
|             ) | ||||
|         } | ||||
|  | ||||
|         if(errors.noErrors()) { | ||||
|             varsPreferringZp.forEach { variable -> | ||||
|                 val result = zeropage.allocate( | ||||
|                     variable.scopedNameString, | ||||
|                     variable.dt, | ||||
|                     variable.length?.toInt(), | ||||
|                     variable.astNode?.position ?: Position.DUMMY, | ||||
|                     errors | ||||
|                 ) | ||||
|                 result.onSuccess { numVariablesAllocatedInZP++ } | ||||
|                 //  no need to check for allocation error, if there is one, just allocate in normal system ram. | ||||
|             } | ||||
|  | ||||
|             // 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? ...? | ||||
|             if(errors.noErrors()) { | ||||
|                 val sortedList = varsDontCareWithoutAlignment.sortedByDescending { it.scopedNameString } | ||||
|                 for (variable in sortedList) { | ||||
|                     if(variable.dt.isIntegerOrBool || variable.dt.isPointer) { | ||||
|                         if(zeropage.free.isEmpty()) { | ||||
|                             break | ||||
|                         } else { | ||||
|                             val result = zeropage.allocate( | ||||
|                                 variable.scopedNameString, | ||||
|                                 variable.dt, | ||||
|                                 variable.length?.toInt(), | ||||
|                                 variable.astNode?.position ?: Position.DUMMY, | ||||
|                                 errors | ||||
|                             ) | ||||
|                             result.onSuccess { numVariablesAllocatedInZP++ } | ||||
|                         } | ||||
|                     } else | ||||
|                         numberOfNonIntegerVariables++ | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         // note: no zeropage allocation is done at all for the @nozp variables. This means they will always end up outside the zeropage. | ||||
|     } | ||||
|  | ||||
|     private fun collectAllVariables(st: SymbolTable): Collection<StStaticVariable> { | ||||
|         val vars = mutableListOf<StStaticVariable>() | ||||
|         fun collect(node: StNode) { | ||||
|             for(child in node.children) { | ||||
|                 if(child.value.type == StNodeType.STATICVAR) | ||||
|                     vars.add(child.value as StStaticVariable) | ||||
|                 else | ||||
|                     collect(child.value) | ||||
|             } | ||||
|         } | ||||
|         collect(st) | ||||
|         return vars.sortedBy { it.dt.base } | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,215 @@ | ||||
| package prog8.codegen.cpu6502.assignment | ||||
|  | ||||
| import prog8.code.ast.PtBinaryExpression | ||||
| import prog8.code.ast.PtExpression | ||||
| import prog8.code.core.AssemblyError | ||||
| import prog8.code.core.ComparisonOperators | ||||
| import prog8.code.core.DataType | ||||
| import prog8.code.core.RegisterOrPair | ||||
| import prog8.code.target.C64Target | ||||
| import prog8.code.target.Cx16Target | ||||
| import prog8.codegen.cpu6502.AsmGen6502Internal | ||||
|  | ||||
| // | ||||
| // This contains codegen for stack-based evaluation of binary expressions. | ||||
| // It uses the CPU stack so depth is limited. | ||||
| // It is called "as a last resort" if the optimized codegen path is unable | ||||
| // to come up with a special case of the expression. | ||||
| // | ||||
| internal class AnyExprAsmGen( | ||||
|     private val asmgen: AsmGen6502Internal | ||||
| ) { | ||||
|     fun assignAnyExpressionUsingStack(expr: PtBinaryExpression, assign: AsmAssignment): Boolean { | ||||
|         if(expr.operator==".") | ||||
|             throw AssemblyError("pointer deref expression should have been handled elsewhere ${expr.position}") | ||||
|         when { | ||||
|             expr.type.isByteOrBool -> { | ||||
|                 if(expr.left.type.isByteOrBool && expr.right.type.isByteOrBool) | ||||
|                     return assignByteBinExpr(expr, assign) | ||||
|                 if (expr.left.type.isWord && expr.right.type.isWord) { | ||||
|                     require(expr.operator in ComparisonOperators) | ||||
|                     throw AssemblyError("words operands comparison -> byte, should have been handled elsewhere") | ||||
|                 } | ||||
|                 if (expr.left.type.isFloat && expr.right.type.isFloat) { | ||||
|                     require(expr.operator in ComparisonOperators) | ||||
|                     return assignFloatBinExpr(expr, assign) | ||||
|                 } | ||||
|                 throw AssemblyError("weird expr operand types: ${expr.left.type} and ${expr.right.type}") | ||||
|             } | ||||
|             expr.type.isWord -> { | ||||
|                 require(expr.left.type.isWord && expr.right.type.isWord) { | ||||
|                     "both operands must be words" | ||||
|                 } | ||||
|                 throw AssemblyError("expression should have been handled otherwise: word ${expr.operator} at ${expr.position}") | ||||
|             } | ||||
|             expr.type.isFloat -> { | ||||
|                 require(expr.left.type.isFloat && expr.right.type.isFloat) { | ||||
|                     "both operands must be floats" | ||||
|                 } | ||||
|                 return assignFloatBinExpr(expr, assign) | ||||
|             } | ||||
|             expr.type.isPointer -> { | ||||
|                 require((expr.left.type.isPointer || expr.left.type.isUnsignedWord) && (expr.right.type.isPointer || expr.right.type.isUnsignedWord)) { | ||||
|                     "both operands must be pointers or uwords" | ||||
|                 } | ||||
|                 throw AssemblyError("expression should have been handled otherwise: pointer ${expr.operator} at ${expr.position}") | ||||
|             } | ||||
|             else -> throw AssemblyError("weird expression type in assignment") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun assignByteBinExpr(expr: PtBinaryExpression, assign: AsmAssignment): Boolean { | ||||
|         when(expr.operator) { | ||||
|             "+" -> { | ||||
|                 asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A) | ||||
|                 asmgen.out("  pha") | ||||
|                 asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE) | ||||
|                 asmgen.out("  pla |  clc |  adc  P8ZP_SCRATCH_B1") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "-" -> { | ||||
|                 asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A) | ||||
|                 asmgen.out("  pha") | ||||
|                 asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE) | ||||
|                 asmgen.out("  pla |  sec |  sbc  P8ZP_SCRATCH_B1") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 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}") | ||||
|             "and" -> TODO("logical and (with optional shortcircuit) ${expr.position}") | ||||
|             "or" -> TODO("logical or (with optional shortcircuit) ${expr.position}") | ||||
|             "&" -> { | ||||
|                 asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A) | ||||
|                 asmgen.out("  pha") | ||||
|                 asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE) | ||||
|                 asmgen.out("  pla |  and  P8ZP_SCRATCH_B1") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "|" -> { | ||||
|                 asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A) | ||||
|                 asmgen.out("  pha") | ||||
|                 asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE) | ||||
|                 asmgen.out("  pla |  ora  P8ZP_SCRATCH_B1") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "^", "xor" -> { | ||||
|                 asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.A) | ||||
|                 asmgen.out("  pha") | ||||
|                 asmgen.assignExpressionToVariable(expr.right, "P8ZP_SCRATCH_B1", DataType.UBYTE) | ||||
|                 asmgen.out("  pla |  eor  P8ZP_SCRATCH_B1") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 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}") | ||||
|             else -> return false | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun assignFloatBinExpr(expr: PtBinaryExpression, assign: AsmAssignment): Boolean { | ||||
|         when(expr.operator) { | ||||
|             "+" -> { | ||||
|                 assignFloatOperandsToFACandARG(expr.left, expr.right) | ||||
|                 asmgen.out("  jsr  floats.FADDT") | ||||
|                 asmgen.assignRegister(RegisterOrPair.FAC1, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "-" -> { | ||||
|                 assignFloatOperandsToFACandARG(expr.right, expr.left) | ||||
|                 asmgen.out("  jsr  floats.FSUBT") | ||||
|                 asmgen.assignRegister(RegisterOrPair.FAC1, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "*" -> { | ||||
|                 assignFloatOperandsToFACandARG(expr.left, expr.right) | ||||
|                 asmgen.out("  jsr  floats.FMULTT") | ||||
|                 asmgen.assignRegister(RegisterOrPair.FAC1, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "/" -> { | ||||
|                 assignFloatOperandsToFACandARG(expr.right, expr.left) | ||||
|                 asmgen.out("  jsr  floats.FDIVT") | ||||
|                 asmgen.assignRegister(RegisterOrPair.FAC1, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "==" -> { | ||||
|                 setupFloatComparisonFAC1vsVarAY(expr) | ||||
|                 asmgen.out("  jsr  floats.var_fac1_equal_f") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "!=" -> { | ||||
|                 setupFloatComparisonFAC1vsVarAY(expr) | ||||
|                 asmgen.out("  jsr  floats.var_fac1_notequal_f") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "<" -> { | ||||
|                 setupFloatComparisonFAC1vsVarAY(expr) | ||||
|                 asmgen.out("  jsr  floats.var_fac1_less_f") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             ">" -> { | ||||
|                 setupFloatComparisonFAC1vsVarAY(expr) | ||||
|                 asmgen.out("  jsr  floats.var_fac1_greater_f") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             "<=" -> { | ||||
|                 setupFloatComparisonFAC1vsVarAY(expr) | ||||
|                 asmgen.out("  jsr  floats.var_fac1_lesseq_f") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             ">=" -> { | ||||
|                 setupFloatComparisonFAC1vsVarAY(expr) | ||||
|                 asmgen.out("  jsr  floats.var_fac1_greatereq_f") | ||||
|                 asmgen.assignRegister(RegisterOrPair.A, assign.target) | ||||
|                 return true | ||||
|             } | ||||
|             else -> TODO("float expression operator ${expr.operator}") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun assignFloatOperandsToFACandARG(left: PtExpression, right: PtExpression) { | ||||
|         when(asmgen.options.compTarget.name) { | ||||
|             C64Target.NAME -> { | ||||
|                 // C64 math library has a quirk: you have always make sure FAC2/ARG is loaded last (done using CONUPK) | ||||
|                 // otherwise the result of certain floating point operations such as FDIVT will be wrong. | ||||
|                 // see https://www.c64-wiki.com/wiki/CONUPK | ||||
|                 // Unfortunately this means we have to push and pop an intermediary floating point value to and from memory. | ||||
|                 asmgen.assignExpressionToRegister(right, RegisterOrPair.FAC1, true) | ||||
|                 asmgen.pushFAC1() | ||||
|                 asmgen.assignExpressionToRegister(left, RegisterOrPair.FAC1, true) | ||||
|                 asmgen.popFAC2() | ||||
|             } | ||||
|             Cx16Target.NAME -> { | ||||
|                 asmgen.assignExpressionToRegister(left, RegisterOrPair.FAC1, true) | ||||
|                 if (!right.isSimple()) asmgen.pushFAC1() | ||||
|                 asmgen.assignExpressionToRegister(right, RegisterOrPair.FAC2, true) | ||||
|                 if (!right.isSimple()) asmgen.popFAC1() | ||||
|             } | ||||
|             else -> TODO("don't know how to evaluate float expression for selected compilation target  ${left.position}") | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     private fun setupFloatComparisonFAC1vsVarAY(expr: PtBinaryExpression) { | ||||
|         asmgen.assignExpressionToRegister(expr.left, RegisterOrPair.FAC1, true) | ||||
|         if(!expr.right.isSimple()) asmgen.pushFAC1() | ||||
|         asmgen.assignExpressionToVariable(expr.right, "floats.floats_temp_var", DataType.FLOAT) | ||||
|         if(!expr.right.isSimple()) asmgen.popFAC1() | ||||
|         asmgen.out("  lda  #<floats.floats_temp_var |  ldy  #>floats.floats_temp_var") | ||||
|     } | ||||
| } | ||||
| @@ -0,0 +1,286 @@ | ||||
| package prog8.codegen.cpu6502.assignment | ||||
|  | ||||
| import prog8.code.ast.* | ||||
| import prog8.code.core.* | ||||
| import prog8.codegen.cpu6502.AsmGen6502Internal | ||||
|  | ||||
|  | ||||
| internal enum class TargetStorageKind { | ||||
|     VARIABLE,       // non-pointer variable | ||||
|     ARRAY, | ||||
|     MEMORY, | ||||
|     REGISTER, | ||||
|     POINTER,        // wherever the pointer variable points to | ||||
|     VOID            // assign nothing - used in multi-value assigns for void placeholders | ||||
| } | ||||
|  | ||||
| internal enum class SourceStorageKind { | ||||
|     LITERALBOOLEAN, | ||||
|     LITERALNUMBER, | ||||
|     VARIABLE, | ||||
|     ARRAY, | ||||
|     MEMORY, | ||||
|     REGISTER, | ||||
|     EXPRESSION,         // expression in ast-form, still to be evaluated | ||||
| } | ||||
|  | ||||
| internal class AsmAssignTarget(val kind: TargetStorageKind, | ||||
|                                private val asmgen: AsmGen6502Internal, | ||||
|                                val datatype: DataType, | ||||
|                                val scope: IPtSubroutine?, | ||||
|                                val position: Position, | ||||
|                                private val variableAsmName: String? = null, | ||||
|                                val array: PtArrayIndexer? = null, | ||||
|                                val memory: PtMemoryByte? = null, | ||||
|                                val register: RegisterOrPair? = null, | ||||
|                                val pointer: PtPointerDeref? = null, | ||||
|                                val origAstTarget: PtAssignTarget? = null | ||||
|                                ) | ||||
| { | ||||
|     val constArrayIndexValue by lazy { array?.index?.asConstInteger()?.toUInt() } | ||||
|     val asmVarname: String by lazy { | ||||
|         if (array == null) | ||||
|             variableAsmName!! | ||||
|         else { | ||||
|             if(array.variable==null) | ||||
|                 TODO("asmVarname for array with pointer") | ||||
|             asmgen.asmVariableName(array.variable!!) | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     init { | ||||
|         if(register!=null && !datatype.isNumericOrBool) | ||||
|             throw AssemblyError("must be numeric type") | ||||
|         if(kind==TargetStorageKind.REGISTER) | ||||
|             require(register!=null) | ||||
|         else | ||||
|             require(register==null) | ||||
|         if(kind==TargetStorageKind.POINTER) | ||||
|             require(pointer!=null) | ||||
|         if(pointer!=null) | ||||
|             require(kind==TargetStorageKind.POINTER) | ||||
|     } | ||||
|  | ||||
|     companion object { | ||||
|         fun fromAstAssignmentMulti(targets: List<PtAssignTarget>, definingSub: IPtSubroutine?, asmgen: AsmGen6502Internal): List<AsmAssignTarget> { | ||||
|             return targets.map { | ||||
|                 if(it.void) | ||||
|                     AsmAssignTarget(TargetStorageKind.VOID, asmgen, DataType.UNDEFINED, null, it.position) | ||||
|                 else | ||||
|                     fromAstAssignment(it, definingSub, asmgen) | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fun fromAstAssignment(target: PtAssignTarget, definingSub: IPtSubroutine?, asmgen: AsmGen6502Internal): AsmAssignTarget { | ||||
|             with(target) { | ||||
|                 when { | ||||
|                     identifier != null -> { | ||||
|                         val parameter = asmgen.findSubroutineParameter(identifier!!.name, asmgen) | ||||
|                         if (parameter!=null) { | ||||
|                             val sub = parameter.definingAsmSub() | ||||
|                             if (sub!=null) { | ||||
|                                 val reg = sub.parameters.single { it.second===parameter }.first | ||||
|                                 if(reg.statusflag!=null) | ||||
|                                     throw AssemblyError("can't assign value to processor statusflag directly") | ||||
|                                 else | ||||
|                                     return AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, type, definingSub, target.position, register=reg.registerOrPair, origAstTarget = this) | ||||
|                             } | ||||
|                         } | ||||
|                         return AsmAssignTarget(TargetStorageKind.VARIABLE, asmgen, type, definingSub, target.position, variableAsmName = asmgen.asmVariableName(identifier!!), origAstTarget =  this) | ||||
|                     } | ||||
|                     array != null -> return AsmAssignTarget(TargetStorageKind.ARRAY, asmgen, type, definingSub, target.position, array = array, origAstTarget =  this) | ||||
|                     memory != null -> return AsmAssignTarget(TargetStorageKind.MEMORY, asmgen, type, definingSub, target.position, memory =  memory, origAstTarget =  this) | ||||
|                     pointerDeref != null -> return AsmAssignTarget(TargetStorageKind.POINTER, asmgen, type, definingSub, target.position, pointer = pointerDeref, origAstTarget =  this) | ||||
|                     else -> throw AssemblyError("weird target") | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         fun fromRegisters(registers: RegisterOrPair, signed: Boolean, pos: Position, scope: IPtSubroutine?, asmgen: AsmGen6502Internal): AsmAssignTarget = | ||||
|                 when(registers) { | ||||
|                     RegisterOrPair.A, | ||||
|                     RegisterOrPair.X, | ||||
|                     RegisterOrPair.Y -> { | ||||
|                         val dt = if(signed) DataType.BYTE else DataType.UBYTE | ||||
|                         AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, dt, scope, pos, register = registers) | ||||
|                     } | ||||
|                     RegisterOrPair.AX, | ||||
|                     RegisterOrPair.AY, | ||||
|                     RegisterOrPair.XY -> { | ||||
|                         val dt = if(signed) DataType.WORD else DataType.UWORD | ||||
|                         AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, dt, scope, pos, register = registers) | ||||
|                     } | ||||
|                     RegisterOrPair.FAC1, | ||||
|                     RegisterOrPair.FAC2 -> { | ||||
|                         AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, DataType.FLOAT, scope, pos, register = registers) | ||||
|                     } | ||||
|                     RegisterOrPair.R0, | ||||
|                     RegisterOrPair.R1, | ||||
|                     RegisterOrPair.R2, | ||||
|                     RegisterOrPair.R3, | ||||
|                     RegisterOrPair.R4, | ||||
|                     RegisterOrPair.R5, | ||||
|                     RegisterOrPair.R6, | ||||
|                     RegisterOrPair.R7, | ||||
|                     RegisterOrPair.R8, | ||||
|                     RegisterOrPair.R9, | ||||
|                     RegisterOrPair.R10, | ||||
|                     RegisterOrPair.R11, | ||||
|                     RegisterOrPair.R12, | ||||
|                     RegisterOrPair.R13, | ||||
|                     RegisterOrPair.R14, | ||||
|                     RegisterOrPair.R15 -> { | ||||
|                         val dt = if(signed) DataType.WORD else DataType.UWORD | ||||
|                         AsmAssignTarget(TargetStorageKind.REGISTER, asmgen, dt, scope, pos, register = registers) | ||||
|                     } | ||||
|                 } | ||||
|     } | ||||
|  | ||||
|     fun isSameAs(left: PtExpression): Boolean = | ||||
|         when(kind) { | ||||
|             TargetStorageKind.VARIABLE -> { | ||||
|                 val scopedName: String = if('.' in asmVarname) | ||||
|                     asmVarname | ||||
|                 else { | ||||
|                     val scopeName = (scope as? PtNamedNode)?.scopedName | ||||
|                     if (scopeName == null) asmVarname else "$scopeName.$asmVarname" | ||||
|                 } | ||||
|                 left is PtIdentifier && left.name==scopedName | ||||
|             } | ||||
|             TargetStorageKind.ARRAY -> { | ||||
|                 left is PtArrayIndexer && left isSameAs array!! && left.splitWords==array.splitWords && (left.pointerderef==null && array.pointerderef==null || left.pointerderef!! isSameAs array.pointerderef!!) | ||||
|             } | ||||
|             TargetStorageKind.MEMORY -> { | ||||
|                 left isSameAs memory!! | ||||
|             } | ||||
|             TargetStorageKind.POINTER -> { | ||||
|                 TODO("is pointer deref target same as expression? ${this.position}") | ||||
|             } | ||||
|             TargetStorageKind.REGISTER -> false | ||||
|             TargetStorageKind.VOID -> false | ||||
|         } | ||||
| } | ||||
|  | ||||
| internal class AsmAssignSource(val kind: SourceStorageKind, | ||||
|                                private val program: PtProgram, | ||||
|                                private val asmgen: AsmGen6502Internal, | ||||
|                                val datatype: DataType, | ||||
|                                private val variableAsmName: String? = null, | ||||
|                                val array: PtArrayIndexer? = null, | ||||
|                                val memory: PtMemoryByte? = null, | ||||
|                                val register: RegisterOrPair? = null, | ||||
|                                val number: PtNumber? = null, | ||||
|                                val boolean: PtBool? = null, | ||||
|                                val expression: PtExpression? = null | ||||
| ) | ||||
| { | ||||
|     val asmVarname: String | ||||
|         get() = if(array==null) | ||||
|             variableAsmName!! | ||||
|         else { | ||||
|             if(array.variable==null) | ||||
|                 TODO("asmVarname for array with pointer") | ||||
|             asmgen.asmVariableName(array.variable!!) | ||||
|         } | ||||
|  | ||||
|     companion object { | ||||
|         fun fromAstSource(value: PtExpression, program: PtProgram, asmgen: AsmGen6502Internal): AsmAssignSource { | ||||
|             val cv = value as? PtNumber | ||||
|             if(cv!=null) | ||||
|                 return AsmAssignSource(SourceStorageKind.LITERALNUMBER, program, asmgen, cv.type, number = cv) | ||||
|             val bv = value as? PtBool | ||||
|             if(bv!=null) | ||||
|                 return AsmAssignSource(SourceStorageKind.LITERALBOOLEAN, program, asmgen, DataType.BOOL, boolean = bv) | ||||
|  | ||||
|             return when(value) { | ||||
|                 // checked above:   is PtNumber -> throw AssemblyError("should have been constant value") | ||||
|                 is PtString -> throw AssemblyError("string literal value should not occur anymore for asm generation") | ||||
|                 is PtArray -> throw AssemblyError("array literal value should not occur anymore for asm generation") | ||||
|                 is PtIdentifier -> { | ||||
|                     val parameter = asmgen.findSubroutineParameter(value.name, asmgen) | ||||
|                     if(parameter?.definingAsmSub() != null) | ||||
|                         throw AssemblyError("can't assign from a asmsub register parameter $value ${value.position}") | ||||
|                     val varName=asmgen.asmVariableName(value) | ||||
|                     // special case: "cx16.r[0-15]" are 16-bits virtual registers of the commander X16 system | ||||
|                     if(value.type.isUnsignedWord && varName.lowercase().startsWith("cx16.r")) { | ||||
|                         val regStr = varName.lowercase().substring(5) | ||||
|                         val reg = RegisterOrPair.valueOf(regStr.uppercase()) | ||||
|                         AsmAssignSource(SourceStorageKind.REGISTER, program, asmgen, value.type, register = reg) | ||||
|                     } else { | ||||
|                         AsmAssignSource(SourceStorageKind.VARIABLE, program, asmgen, value.type, variableAsmName = varName) | ||||
|                     } | ||||
|                 } | ||||
|                 is PtMemoryByte -> { | ||||
|                     AsmAssignSource(SourceStorageKind.MEMORY, program, asmgen, DataType.UBYTE, memory = value) | ||||
|                 } | ||||
|                 is PtArrayIndexer -> { | ||||
|                     AsmAssignSource(SourceStorageKind.ARRAY, program, asmgen, value.type, array = value) | ||||
|                 } | ||||
|                 is PtBuiltinFunctionCall -> { | ||||
|                     AsmAssignSource(SourceStorageKind.EXPRESSION, program, asmgen, value.type, expression = value) | ||||
|                 } | ||||
|                 is PtFunctionCall -> { | ||||
|                     val symbol = asmgen.symbolTable.lookup(value.name) ?: throw AssemblyError("lookup error ${value.name}") | ||||
|                     val sub = symbol.astNode as IPtSubroutine | ||||
|                     val returnType = | ||||
|                         if(sub is PtSub && sub.signature.returns.size>1) | ||||
|                             DataType.UNDEFINED      // TODO list of types instead? | ||||
|                         else | ||||
|                             sub.returnsWhatWhere().firstOrNull { rr -> rr.first.registerOrPair != null || rr.first.statusflag!=null }?.second | ||||
|                             ?: throw AssemblyError("can't translate zero return values in assignment") | ||||
|                     AsmAssignSource(SourceStorageKind.EXPRESSION, program, asmgen, returnType, expression = value) | ||||
|                 } | ||||
|                 else -> { | ||||
|                     AsmAssignSource(SourceStorageKind.EXPRESSION, program, asmgen, value.type, expression = value) | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fun adjustSignedUnsigned(target: AsmAssignTarget): AsmAssignSource { | ||||
|         // allow some signed/unsigned relaxations | ||||
|  | ||||
|         fun withAdjustedDt(newType: DataType) = | ||||
|                 AsmAssignSource(kind, program, asmgen, newType, variableAsmName, array, memory, register, number, boolean, expression) | ||||
|  | ||||
|         if(target.datatype!=datatype) { | ||||
|             if(target.datatype.isByte && datatype.isByte) { | ||||
|                 return withAdjustedDt(target.datatype) | ||||
|             } else if(target.datatype.isWord && datatype.isWord) { | ||||
|                 return withAdjustedDt(target.datatype) | ||||
|             } | ||||
|         } | ||||
|         return this | ||||
|     } | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
| internal sealed class AsmAssignmentBase(val source: AsmAssignSource, | ||||
|                                         val targets: List<AsmAssignTarget>, | ||||
|                                         val memsizer: IMemSizer, | ||||
|                                         val position: Position) { | ||||
|     init { | ||||
|         targets.forEach { target -> | ||||
|             if (!source.datatype.isArray && !source.datatype.isUndefined && !target.datatype.isArray && !target.datatype.isUndefined) | ||||
|                 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}" | ||||
|                 } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     val target: AsmAssignTarget | ||||
|         get() = targets.single() | ||||
| } | ||||
|  | ||||
| internal class AsmAssignment(source: AsmAssignSource, | ||||
|                              targets: List<AsmAssignTarget>, | ||||
|                              memsizer: IMemSizer, | ||||
|                              position: Position): AsmAssignmentBase(source, targets, memsizer, position) | ||||
|  | ||||
| internal class AsmAugmentedAssignment(source: AsmAssignSource, | ||||
|                                       val operator: String, | ||||
|                                       target: AsmAssignTarget, | ||||
|                                       memsizer: IMemSizer, | ||||
|                                       position: Position): AsmAssignmentBase(source, listOf(target), memsizer, position) | ||||
|  | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user