mirror of
				https://github.com/davidgiven/fluxengine.git
				synced 2025-10-24 11:11:02 -07:00 
			
		
		
		
	Compare commits
	
		
			891 Commits
		
	
	
		
			FluxEngine
			...
			libui
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | f9b7c7f6c1 | ||
|  | 855d346b8a | ||
|  | 473e81a2da | ||
|  | d925391d57 | ||
|  | 63a5954dfa | ||
|  | 897931f273 | ||
|  | 03e80b2f1c | ||
|  | 53c8ec864c | ||
|  | 5d3002f118 | ||
|  | 5eeb52660c | ||
|  | 3dfafaa278 | ||
|  | 462bd9ae5e | ||
|  | dab0fcc7c0 | ||
|  | b8a3e8085e | ||
|  | c9d1d72ba3 | ||
|  | ccf5d513d2 | ||
|  | d157b7b05d | ||
|  | 05981ac46c | ||
|  | 08615a5954 | ||
|  | b6dbe26c7e | ||
|  | 8a460b3e56 | ||
|  | 012f1a872b | ||
|  | 4f0d61b379 | ||
|  | 9e5e494f88 | ||
|  | b15fd05e8d | ||
|  | faabc28d1b | ||
|  | 4b815846ee | ||
|  | fe8be18c4c | ||
|  | 519c30321d | ||
|  | 77be88a5bb | ||
|  | 8d04931f9f | ||
|  | 3d1ee7a43e | ||
|  | 2584a25527 | ||
|  | 0647a82d38 | ||
|  | c56459c98c | ||
|  | 3347d057fe | ||
|  | 7b0a4a057d | ||
|  | eb0bb2cc2f | ||
|  | 5ff7651218 | ||
|  | dcb3761c9b | ||
|  | 6fa83f4dcb | ||
|  | 001c65dba8 | ||
|  | 969a08bd9a | ||
|  | 3c3e520594 | ||
|  | ed9ffc9091 | ||
|  | 5d0750e835 | ||
|  | 3283d6e8c0 | ||
|  | 1b96f60b2a | ||
|  | a15d78c040 | ||
|  | 40b7f57116 | ||
|  | 68314cd44d | ||
|  | 17787b97d4 | ||
|  | da84297b2c | ||
|  | e3e3bb770d | ||
|  | 9d4f180741 | ||
|  | 91281bb58b | ||
|  | 4ddd7d04b5 | ||
|  | 316836d9f6 | ||
|  | aa3b0a117a | ||
|  | 0f7df3281f | ||
|  | 4a91a35799 | ||
|  | 36c2263675 | ||
|  | 28068d8d31 | ||
|  | 97cc54a07a | ||
|  | 0c3c08db36 | ||
|  | bb5d62fe69 | ||
|  | 376270dd53 | ||
|  | 9056b23eaa | ||
|  | 634be4ae51 | ||
|  | 99901f365e | ||
|  | 422620f4fe | ||
|  | dd3cb93ed9 | ||
|  | dbb422a410 | ||
|  | 68712b8be7 | ||
|  | da7e2b1b54 | ||
|  | 8cff85ad3d | ||
|  | 9ffde65bb0 | ||
|  | 5759bc602f | ||
|  | 4bb6fdc79a | ||
|  | bd019d5d70 | ||
|  | 0457282d47 | ||
|  | 06d8609200 | ||
|  | fa975b5511 | ||
|  | 194b9b1193 | ||
|  | 27fceb3f41 | ||
|  | 930e0bcd67 | ||
|  | d81e0a3ed4 | ||
|  | ebb5c17be4 | ||
|  | 1e99fe5b04 | ||
|  | 5f5f22c82b | ||
|  | 709e300960 | ||
|  | d2f677d84e | ||
|  | 18d90c44dd | ||
|  | 298f77f52e | ||
|  | 70b2f7efb4 | ||
|  | 78a070bd99 | ||
|  | c88ff973b1 | ||
|  | 66f82f764d | ||
|  | d36a18c17a | ||
|  | 8efd95346a | ||
|  | 04b91377c9 | ||
|  | f5d04d989c | ||
|  | 06a497f346 | ||
|  | b1cf706a8f | ||
|  | 91f04e36e9 | ||
|  | 5d76eec0f6 | ||
|  | 7822325866 | ||
|  | 521123f139 | ||
|  | b3f511b025 | ||
|  | ff63840beb | ||
|  | 1fbcf4aa5f | ||
|  | 8d6a811204 | ||
|  | 08fe2a9e27 | ||
|  | d8465f4374 | ||
|  | 8a3cd14723 | ||
|  | 91e794add1 | ||
|  | 9c98caf21d | ||
|  | f97c42017f | ||
|  | 3033a2cb95 | ||
|  | bcf6f48d46 | ||
|  | 4c97f15a65 | ||
|  | 5dc6349818 | ||
|  | 6ab5c4012a | ||
|  | 1d50e45777 | ||
|  | 3cbf420acd | ||
|  | ea407b2182 | ||
|  | 5228885760 | ||
|  | 676845aaf3 | ||
|  | 61af9b3810 | ||
|  | 406a433c3f | ||
|  | 2abd413c08 | ||
|  | ad96c9bb9f | ||
|  | b412d54998 | ||
|  | 6247d3c5e6 | ||
|  | b7ee513dfd | ||
|  | 3795abc19e | ||
|  | 1d32a4d984 | ||
|  | a20a78cd64 | ||
|  | a2f51b36ec | ||
|  | 910ccb273a | ||
|  | 2cbe39e553 | ||
|  | 994c8fc02d | ||
|  | 57dcd6d520 | ||
|  | 20ade1de7b | ||
|  | 28aff78469 | ||
|  | 056055bf0b | ||
|  | 368d329459 | ||
|  | 2fe1ffeaf1 | ||
|  | e252e8eb1d | ||
|  | e160bcc7d5 | ||
|  | 41bd035690 | ||
|  | b0003fc7e5 | ||
|  | 8e1be97589 | ||
|  | 181f2f38d8 | ||
|  | 661940114e | ||
|  | 9874d4bec5 | ||
|  | 7d5fedf35f | ||
|  | 69ad36a9ae | ||
|  | a00137d742 | ||
|  | e002842640 | ||
|  | 41e9c46cba | ||
|  | bc2d58bee3 | ||
|  | c54de27503 | ||
|  | d9bfd77fba | ||
|  | 336cc0077c | ||
|  | 0be673b210 | ||
|  | 028e2498fb | ||
|  | 249f1dea9d | ||
|  | 07c7b22669 | ||
|  | 6623058a18 | ||
|  | 68e4d569d0 | ||
|  | 7186b25a69 | ||
|  | 42cacb18bd | ||
|  | d09c03f4d2 | ||
|  | ad240b150e | ||
|  | 1177ef6f8d | ||
|  | 2ff50c0c56 | ||
|  | 4f4ed1307f | ||
|  | 6d5b7cb64e | ||
|  | 00fc4e3890 | ||
|  | 7344ee4402 | ||
|  | 740eacc7ac | ||
|  | 8b1bcf21ee | ||
|  | b0a5174c0a | ||
|  | b7cca1b95b | ||
|  | 44c51f1246 | ||
|  | 516d43d7a8 | ||
|  | 86d49d563e | ||
|  | a2911a9585 | ||
|  | 9a12b651f9 | ||
|  | 9fa631acca | ||
|  | 8df7998a83 | ||
|  | c81c1926c0 | ||
|  | 7ab1288424 | ||
|  | 3ee88adfa9 | ||
|  | 29e8c99b4f | ||
|  | 8b115f8156 | ||
|  | 11c546f113 | ||
|  | c087539eb7 | ||
|  | 27f7cbb892 | ||
|  | 11ffb09b66 | ||
|  | 114f9f522d | ||
|  | 91dc51d59a | ||
|  | 11b3922b02 | ||
|  | 05552cc3e5 | ||
|  | 3db7964509 | ||
|  | 2a06adcabb | ||
|  | 8cec3354ea | ||
|  | f9df728f45 | ||
|  | 46eead34c4 | ||
|  | 786b35fee2 | ||
|  | 5c6c609991 | ||
|  | 0be8fedf26 | ||
|  | 1f9aaf1ac0 | ||
|  | 3926de3fa1 | ||
|  | 9086f18413 | ||
|  | ad2576bc76 | ||
|  | 4523407d9d | ||
|  | 53162ac934 | ||
|  | 1dac51d4ba | ||
|  | 13deef3416 | ||
|  | b887bebb26 | ||
|  | d51160babb | ||
|  | 44b9e7a398 | ||
|  | 065cd113c1 | ||
|  | a5f618b7e4 | ||
|  | a06e4e862d | ||
|  | 084c858446 | ||
|  | 5d889d4d95 | ||
|  | f589e094b2 | ||
|  | 2eff798d74 | ||
|  | 239cecff9e | ||
|  | fa36af454e | ||
|  | 458d0f7a1b | ||
|  | f1f27ffd33 | ||
|  | 4a2e09e8eb | ||
|  | 4b3fada646 | ||
|  | 690befc98b | ||
|  | c6b3c0f9eb | ||
|  | b423a71b38 | ||
|  | e226773066 | ||
|  | 98918d160a | ||
|  | 969298fb58 | ||
|  | bc60f3b45a | ||
|  | f7a4785d22 | ||
|  | 1bf41cbfd7 | ||
|  | e9d80423ae | ||
|  | 928ffbd7af | ||
|  | 25ebad2448 | ||
|  | dd951dc0ed | ||
|  | aed9e44b6b | ||
|  | 1fa2547aff | ||
|  | 164ceb845e | ||
|  | ce463686dc | ||
|  | 92aa28cac2 | ||
|  | f8674230ed | ||
|  | 326969e488 | ||
|  | 155e8c2a4b | ||
|  | 7f95f6b43f | ||
|  | b2d7ba1a65 | ||
|  | b550bbbd08 | ||
|  | cc79977ac0 | ||
|  | 4cbb75df60 | ||
|  | 7d8926659e | ||
|  | 37ca3f74c7 | ||
|  | 56cbf39d59 | ||
|  | a783e7f500 | ||
|  | 2f4ac42684 | ||
|  | ec689e100e | ||
|  | 6f0909d992 | ||
|  | 0092dec49e | ||
|  | 3447689c19 | ||
|  | d4563fd842 | ||
|  | d2ecec6009 | ||
|  | 26f26aec50 | ||
|  | 34dfc2767f | ||
|  | bc3d3cabce | ||
|  | 23d80b93ae | ||
|  | 0d59d3d195 | ||
|  | 192427cf80 | ||
|  | 98e4094d70 | ||
|  | e88b939866 | ||
|  | aed5a02ee1 | ||
|  | 9f1e1bb2b6 | ||
|  | 15e9383d0b | ||
|  | 4ec056900c | ||
|  | 983feb59b0 | ||
|  | 8b2ce33f83 | ||
|  | 44b452b30b | ||
|  | f60c9981b8 | ||
|  | 7c3df93580 | ||
|  | 601d7dfcf8 | ||
|  | 41cb53a550 | ||
|  | 87c13ae20c | ||
|  | c30482af66 | ||
|  | 5f4f2f10d7 | ||
|  | ae630a0e18 | ||
|  | c66931cb12 | ||
|  | fb2dbe25cd | ||
|  | bf4831be9f | ||
|  | 0f83082cf0 | ||
|  | fbfde604e7 | ||
|  | 5e2dfbed73 | ||
|  | 571602d3de | ||
|  | 85693f2577 | ||
|  | 60bc8ad888 | ||
|  | 9bf309eb5c | ||
|  | eeddfa87b6 | ||
|  | 861fe2466c | ||
|  | 3a2cce78ca | ||
|  | 4e01f1de0d | ||
|  | 4f0a5e854c | ||
|  | c669a9c808 | ||
|  | e4d827256f | ||
|  | 1c1ad2725e | ||
|  | d18e54c5a3 | ||
|  | 7a885e23d3 | ||
|  | 33dde3a465 | ||
|  | 8d4ac59f03 | ||
|  | 4fbfc492ce | ||
|  | ac1fb71bc5 | ||
|  | 01d6b37b83 | ||
|  | 2284b4dce1 | ||
|  | b92f80e6be | ||
|  | bd5596fa30 | ||
|  | 4614b63c30 | ||
|  | cf41b6cbb2 | ||
|  | 87f3e0e291 | ||
|  | 1997abcde6 | ||
|  | 8b761298ee | ||
|  | df0356b841 | ||
|  | dc158ebff4 | ||
|  | f8192b90f4 | ||
|  | 5237049e2c | ||
|  | 9d70fb261e | ||
|  | 8370b8a743 | ||
|  | 25656d81a1 | ||
|  | be97791428 | ||
|  | 18b683d22e | ||
|  | 34e9742f71 | ||
|  | 087eb7777e | ||
|  | c469c5e3a1 | ||
|  | 485c29bb49 | ||
|  | aa0cf150bd | ||
|  | 91c52e91cd | ||
|  | a1f4014738 | ||
|  | 8a136ac4f6 | ||
|  | ea43b3dc6a | ||
|  | 57fc787819 | ||
|  | a0164b8de3 | ||
|  | 9df35c1814 | ||
|  | 9d2a5fee86 | ||
|  | a6b2e932fa | ||
|  | 05aaa2634b | ||
|  | 2c98f5c542 | ||
|  | d246fca9df | ||
|  | c79feb405c | ||
|  | b1145f8da3 | ||
|  | 3588d681a2 | ||
|  | 55d894ae1f | ||
|  | 0509caf432 | ||
|  | 5d1d807e78 | ||
|  | 22ba38b2e0 | ||
|  | 42f858267c | ||
|  | 5b1a3173f8 | ||
|  | 97822bd9a8 | ||
|  | b926f5a692 | ||
|  | 191dfdf13f | ||
|  | a100f1fe1e | ||
|  | 471ccf281c | ||
|  | 6c959be188 | ||
|  | 010887f18f | ||
|  | e4b5e4c502 | ||
|  | 2de3b4f92e | ||
|  | 225a93330d | ||
|  | 41b36649a9 | ||
|  | 55c26ab1c4 | ||
|  | 8b1e60a1fd | ||
|  | ed0f38748b | ||
|  | 7d75a720ca | ||
|  | 5a38db166f | ||
|  | 3c9fb79263 | ||
|  | 98d5a2dad9 | ||
|  | 4ab8b4984d | ||
|  | f741ad058e | ||
|  | 2512a4fc32 | ||
|  | 5554d5e608 | ||
|  | 48d5ed2ff9 | ||
|  | 2632668d0e | ||
|  | f46e444aa2 | ||
|  | 1149ad86a2 | ||
|  | e1398d98b0 | ||
|  | 8133e2b7aa | ||
|  | ebe678b18b | ||
|  | 509217606c | ||
|  | 6fb694669c | ||
|  | 5a63172a86 | ||
|  | 93dcc7e242 | ||
|  | 243eea33e9 | ||
|  | 38a8367f62 | ||
|  | a02953cccc | ||
|  | f13f96967e | ||
|  | f7c31281e0 | ||
|  | ac34a43d9b | ||
|  | c8d0950979 | ||
|  | a4ff59eccb | ||
|  | c3a50f9442 | ||
|  | 57e81ee72e | ||
|  | 05df0a37b1 | ||
|  | 25f2c3a8c1 | ||
|  | c3aa12db78 | ||
|  | a3bd7cc644 | ||
|  | 5a186b6960 | ||
|  | 639588fa68 | ||
|  | f9510c54b2 | ||
|  | 3a8ddf8025 | ||
| a2801ea88c | |||
|  | c2aae7ee18 | ||
|  | 9d0804eff4 | ||
|  | 6ff84b3693 | ||
|  | b641e0282b | ||
|  | 37a467cabc | ||
|  | e01a7110ac | ||
|  | 5dad5de548 | ||
|  | 295325a20b | ||
|  | df0a9bac96 | ||
|  | cf9cef6f87 | ||
|  | f92814b24b | ||
|  | 331b59cd1e | ||
|  | ed6d211aff | ||
|  | a8f1469d36 | ||
|  | 74cb332706 | ||
|  | 89165369b1 | ||
|  | a54e3d33a6 | ||
|  | 0f17984f41 | ||
|  | 6527ceb913 | ||
|  | 64a57ba837 | ||
|  | 74da9330f8 | ||
|  | 4fa1dd6860 | ||
|  | e55effc9ca | ||
|  | 924b862f7c | ||
|  | 5f077762d5 | ||
|  | 1b0ec50711 | ||
|  | 22f320f1c4 | ||
|  | 9949476584 | ||
|  | 1b5b170557 | ||
|  | 925a3a4bdb | ||
|  | 720fe9f95f | ||
|  | 7c4f8e1443 | ||
|  | 79d24dff46 | ||
|  | 928435a31d | ||
|  | 26ac92eaa3 | ||
|  | 2974c08b08 | ||
|  | 5a7b0b3050 | ||
|  | cc2d9bbdd1 | ||
|  | e912152784 | ||
|  | d00681f623 | ||
|  | 52942c3d2a | ||
|  | dedabdd8d8 | ||
|  | 3061499860 | ||
|  | 20f18755ed | ||
|  | 333f2aba54 | ||
|  | 1cb673ed80 | ||
|  | 57e0bc784a | ||
|  | 425afa93da | ||
|  | 259b2cebc7 | ||
|  | 06589826c8 | ||
|  | 2245cd982a | ||
|  | 065b50769f | ||
|  | d1e99852bc | ||
|  | bf8f6ae687 | ||
|  | 4ad6805ea1 | ||
|  | 4f4e3f0b89 | ||
|  | b51f2c1ec8 | ||
|  | 1bec06fd75 | ||
|  | 451d2e2d1d | ||
|  | 9cee12f9f4 | ||
|  | f5d6011a77 | ||
|  | 64b2ff19ea | ||
|  | 9c17a64773 | ||
|  | 69c877ff66 | ||
|  | ac557ffedc | ||
|  | b1e41bc583 | ||
|  | a144395ec9 | ||
|  | 0cf9d05489 | ||
|  | 0e6c0a41d0 | ||
|  | 8a83652d08 | ||
|  | 91ee72a8d6 | ||
|  | 91b1c8c13c | ||
|  | 26effeabe6 | ||
|  | 611c9740ed | ||
|  | 2a048c3228 | ||
|  | f4fd83d999 | ||
|  | cf68585808 | ||
|  | 9f9e926cff | ||
|  | 9dc0dd75fd | ||
|  | 9f285710f8 | ||
|  | ee1c448327 | ||
|  | e85bf1713e | ||
|  | 7341c71277 | ||
|  | d579863311 | ||
|  | c79cfc19aa | ||
|  | 997a6be267 | ||
|  | 762bb3006d | ||
|  | 07f2bd8cab | ||
|  | daf83db9b3 | ||
|  | 55c6e19af4 | ||
|  | 9cadc94c5a | ||
|  | cacdf9ef56 | ||
|  | a3042fc6c0 | ||
|  | 3efd492525 | ||
|  | 55a5cbc356 | ||
|  | 2887b024ab | ||
|  | 917303edb9 | ||
|  | c712c15a30 | ||
|  | 0c541db8e0 | ||
|  | 603b1520d7 | ||
|  | c7eb8ad5c8 | ||
|  | 0b285aa7f4 | ||
|  | e8665bd00c | ||
|  | fb4eaa4332 | ||
|  | 874a9eae76 | ||
|  | 8230520956 | ||
|  | 66da9675f1 | ||
|  | 61ff48c005 | ||
|  | 5fc8a1e52a | ||
|  | df1ac74069 | ||
|  | 91f718bf38 | ||
|  | 46e987e393 | ||
|  | a59b4f7be7 | ||
|  | ca66e3c35c | ||
|  | 31e2ad8cba | ||
|  | 320f32895a | ||
|  | d4db131d3c | ||
|  | 27c2c9045e | ||
|  | f97b7c7d62 | ||
|  | 9eb33d31ac | ||
|  | 6dd84d6fc2 | ||
|  | daddd60581 | ||
|  | e832723ee4 | ||
|  | 35f4a63c0e | ||
|  | 28478ea4ac | ||
|  | 0f93a68694 | ||
|  | 1a2d5d13b3 | ||
|  | 5f2894fc5b | ||
|  | 66cb39dce2 | ||
|  | d44c871c54 | ||
|  | dff0378ba8 | ||
|  | 4f7b1b7140 | ||
|  | 662514304b | ||
|  | 0913e9e0c0 | ||
|  | d403733627 | ||
|  | fae5b439d0 | ||
|  | bc66de6d85 | ||
|  | 57e598156c | ||
|  | b570e44ee4 | ||
|  | b115d0b55b | ||
|  | 03fc1419de | ||
|  | 8b71c0d737 | ||
|  | a833aa0a00 | ||
|  | 52332b04ac | ||
|  | 529488cab0 | ||
|  | b2429a7ca3 | ||
|  | 0bce12d3b4 | ||
|  | 75f557cb18 | ||
|  | 035dd1fad1 | ||
|  | d2df79a665 | ||
|  | 103e0a13bb | ||
|  | d1b5eec84a | ||
|  | 6b1e6b31ed | ||
|  | f6f6db913e | ||
|  | ec0a6416fd | ||
|  | 1787402be9 | ||
|  | 5f6d99f138 | ||
|  | d1e2b0d1f8 | ||
|  | c2c51bbe33 | ||
|  | bb806e3853 | ||
|  | a11d0e75c8 | ||
|  | 5406ff0ea3 | ||
|  | c88317b44a | ||
|  | 6898062d66 | ||
|  | 6e1f264e6a | ||
|  | 082be14232 | ||
|  | 231aa44d03 | ||
|  | cdb12f35d4 | ||
|  | e831ee8b44 | ||
|  | 40e9a6082f | ||
|  | 53cec292d0 | ||
|  | 3f85309ee5 | ||
|  | 70944f8521 | ||
|  | 2ab00c42ff | ||
|  | a572742caa | ||
|  | 400e5f8580 | ||
|  | 74f0fd89b6 | ||
|  | 09f9bea7a2 | ||
|  | 8bffb38117 | ||
|  | eb5d545c35 | ||
|  | a79a545730 | ||
|  | 3863dab944 | ||
|  | e53b7ecd8b | ||
|  | 7d88673ed5 | ||
|  | 41f2da71e4 | ||
|  | cb4ee0fd74 | ||
|  | 088381a5a6 | ||
|  | 629af2a697 | ||
|  | 884edfd497 | ||
|  | 83dd9e462e | ||
|  | 70a6dfd98a | ||
|  | 7f5d96382b | ||
|  | fd4d1c4bb7 | ||
|  | 7eaf3de572 | ||
|  | 4b608de3fb | ||
|  | b47e6e852b | ||
|  | a8a8ce4a36 | ||
|  | c61376d5a1 | ||
|  | d3a5bb08d3 | ||
|  | f1506d0dbd | ||
|  | 15e6d4959e | ||
|  | 41216fd1cd | ||
|  | b8786866db | ||
|  | 82bd1bead4 | ||
|  | 6e2bdcad79 | ||
|  | ef3c9f3d03 | ||
|  | 5427f24df2 | ||
|  | b374340303 | ||
|  | c78ed2c6ad | ||
|  | 3b02bc8cf1 | ||
|  | c7e48a7e76 | ||
|  | 77d125c03d | ||
|  | 8aa52aeefd | ||
|  | 0bab038454 | ||
|  | 6c3b49f4d0 | ||
|  | 03dd689f17 | ||
|  | c375c948c0 | ||
|  | cbcf457ce3 | ||
|  | 4855f825e2 | ||
|  | 85bc1637f2 | ||
|  | 73398b83a9 | ||
|  | 2727e66d40 | ||
|  | 8b6be5a501 | ||
|  | 4fee29307c | ||
|  | 35f8249c67 | ||
|  | d1467a14b8 | ||
|  | 3e6b9eb74d | ||
|  | ce2e8fb4b5 | ||
|  | 7eaa75c05d | ||
|  | e86de4483a | ||
|  | 203a74713f | ||
|  | 59ed2a6793 | ||
|  | a03283ce64 | ||
|  | 984cdaeb03 | ||
|  | a1ed4a9171 | ||
|  | 93caf8e549 | ||
|  | 3841942153 | ||
|  | 5706877b67 | ||
|  | d60900262b | ||
|  | 54ea34400b | ||
|  | db2ab8841a | ||
|  | adba93ae0a | ||
|  | 98587d04a7 | ||
|  | 0051b64648 | ||
|  | 603009ba15 | ||
|  | adb9809692 | ||
|  | 06eb10d2a0 | ||
|  | 2244299bd9 | ||
|  | 6ca06ecafb | ||
|  | 9a5958f80b | ||
|  | 2b53ac057c | ||
|  | 5deba8af41 | ||
|  | 3c54a663b8 | ||
|  | 1fd65452c4 | ||
|  | 30646ccb07 | ||
|  | 5be7249a30 | ||
|  | 067af18103 | ||
|  | 8dbd2a72a7 | ||
|  | c29e131a3b | ||
|  | a9e30c1e49 | ||
|  | 972c8c6b61 | ||
|  | 2007ff7546 | ||
|  | 64694580cd | ||
|  | deaab94494 | ||
|  | 1509e1f89d | ||
|  | 29e1ddc2ff | ||
|  | 1fe6434563 | ||
|  | 0367b7e77d | ||
|  | e6da85bf64 | ||
|  | cd19fcdadd | ||
|  | 1954f02cfb | ||
|  | 39b23200b0 | ||
|  | 0644d6d965 | ||
|  | a075694d8e | ||
|  | b1ea5a9a35 | ||
|  | 00087cbb6b | ||
|  | 1b48ea20c4 | ||
|  | 3d0f019fc4 | ||
|  | a08bfc183f | ||
|  | c5aef9b051 | ||
|  | fc2655ecd6 | ||
|  | a737c723d3 | ||
|  | 37aa8b62b0 | ||
|  | a401173f6d | ||
|  | ce76dc4279 | ||
|  | 1025bd857b | ||
|  | 025802b2d0 | ||
|  | adbcb2cd31 | ||
|  | c47a563790 | ||
|  | 04c09d1a5b | ||
|  | 323da8272a | ||
|  | 38700c79fc | ||
|  | d504d1890a | ||
|  | d53e757cfb | ||
|  | 4983239458 | ||
|  | 376985828a | ||
|  | dce0a26820 | ||
|  | 14e0a67e7d | ||
|  | 1656947764 | ||
|  | 647862cdbd | ||
|  | 4a8d83838c | ||
|  | 8acf8e181d | ||
|  | 2df9920209 | ||
|  | 1a6c6b5420 | ||
|  | edc56d44d6 | ||
|  | ef4eff0195 | ||
|  | df8d45bf66 | ||
|  | 89a27619ff | ||
|  | 387a86969a | ||
|  | acb5059d17 | ||
|  | a4002d2617 | ||
|  | a63a90bbd0 | ||
|  | d25f96dd24 | ||
|  | e8febe6508 | ||
|  | ad3a930c6a | ||
|  | be41c1de76 | ||
|  | d528978667 | ||
|  | 827fcf69d2 | ||
|  | 711ff545e0 | ||
|  | 5befa31050 | ||
|  | 8e5c2d0ebb | ||
|  | f95fceeb3d | ||
|  | 003b20dbf0 | ||
|  | cd9bbaa4b6 | ||
|  | 71e622bf72 | ||
|  | 2a065a08df | ||
|  | 6087228378 | ||
|  | efd74e0d7b | ||
|  | b68a9dcc4f | ||
|  | 008855daa9 | ||
|  | 7a9d36de2a | ||
|  | c56e982c9a | ||
|  | 002cc171a2 | ||
|  | 32e721b47a | ||
|  | 1e82f697a9 | ||
|  | fa09631e32 | ||
|  | e06436ce1e | ||
|  | b2f443e1ad | ||
|  | 2e07be0cf7 | ||
|  | bf0b14d094 | ||
|  | c9f5803194 | ||
|  | 5293560c02 | ||
|  | c49823aa9d | ||
|  | c4ef4882ae | ||
|  | a8eca06cf0 | ||
|  | 065257b5aa | ||
|  | 29bdfc043a | ||
|  | 933ffe7ab4 | ||
|  | e517f28563 | ||
|  | 91ffcf59c3 | ||
|  | 51c618f325 | ||
|  | 9dc1067032 | ||
|  | 9e75dc3af1 | ||
|  | efa4c933b3 | ||
|  | 6af80d1e5e | ||
|  | 0c48897814 | ||
|  | 60e5e35947 | ||
|  | 86c4e959ca | ||
|  | b0c675c589 | ||
|  | d77841c3b7 | ||
|  | 4ed1fb6bac | ||
|  | bcc9e9d9a5 | ||
|  | ec327e25a4 | ||
|  | d0ed5b32f7 | ||
|  | 7c66e1b0d4 | ||
|  | 4475e9f085 | ||
|  | 5c9639ec5a | ||
|  | 792cc88192 | ||
|  | 21fe586724 | ||
|  | 5a0fb2761a | ||
|  | ef4581ed39 | ||
|  | 73419704c2 | ||
|  | a8b92d4780 | ||
|  | 98140b0646 | ||
|  | 4429ce1f84 | ||
|  | 1f50941a2c | ||
|  | a7de04848c | ||
|  | c264fec6e9 | ||
|  | 4488b2542f | ||
|  | 2f1a5189d6 | ||
|  | effaeff51e | ||
|  | 1210549f59 | ||
|  | 7200de9702 | ||
|  | 5dd5c8516a | ||
|  | f7fb2a844b | ||
|  | 20b1b2a4a8 | ||
|  | f8b8bc2295 | ||
|  | 2d4d56d09f | ||
|  | 39599b76c8 | ||
|  | c2c40ccfbb | ||
|  | ab42eb23f4 | ||
|  | 05eff0e528 | ||
|  | 23311b4b68 | ||
|  | 5e97df8d15 | ||
|  | 898e8c551c | ||
|  | ad69c6bd27 | ||
|  | 661399cc83 | ||
|  | edbb4b1daa | ||
|  | 6389e8a756 | ||
|  | c187b79d80 | ||
|  | edbe624c5a | ||
|  | 44e2334815 | ||
|  | b448ab7917 | ||
|  | 072a097003 | ||
|  | a66e704bab | ||
|  | ed0d578b18 | ||
|  | 32bb956710 | ||
|  | f436d6b582 | ||
|  | d2f8c27cb6 | ||
|  | eaa3c57425 | ||
|  | 549f12a2ab | ||
|  | aea254fbe7 | ||
|  | 8ee6eed4dc | ||
|  | 3094c5c919 | ||
|  | 1e012699af | ||
|  | 91d6e9aeb9 | ||
|  | a40b26ff46 | ||
|  | ebcb9c4bb0 | ||
|  | 2520834b18 | ||
|  | a1f3087046 | ||
|  | e9670e205e | ||
|  | 658e2b7295 | ||
|  | 7b4a8d6de2 | ||
|  | e8f7b51aef | ||
|  | 9d6bc57a5f | ||
|  | 73766f92b4 | ||
|  | 80badf3b54 | ||
|  | 116529f85a | ||
|  | 5a2b2bc07a | ||
|  | 41070395c0 | ||
|  | 4304d1eede | ||
|  | 46f1b0aef4 | ||
|  | 9923d67a7c | ||
|  | 99335a84fd | ||
|  | c266779433 | ||
|  | bdcc12cd53 | ||
|  | 7988d0fe24 | ||
|  | 27f5c294b1 | ||
|  | b9a53e0d1c | ||
|  | f8b6d5e6fb | ||
|  | 04ff31c348 | ||
|  | 77b4aebd1b | ||
|  | 4056364300 | ||
|  | 60bfe050d3 | ||
|  | 28d0ce765e | ||
|  | 4954d33307 | ||
|  | 55f3354287 | ||
|  | d6ae373fa8 | ||
|  | a626d5f9a0 | ||
|  | 29db67528d | ||
|  | 31d7477c6a | ||
|  | 56af9eaf18 | ||
|  | 5de0636fe7 | ||
|  | f9117b8d11 | ||
|  | 10d385375f | ||
|  | 2f72c3f8f0 | ||
|  | 54edff9b94 | ||
|  | 112377f885 | ||
|  | 87e29fc386 | ||
|  | b1db5c48b1 | ||
|  | 38fab7edcb | ||
|  | d8172154c3 | ||
|  | eb924780ab | ||
|  | 28e0ef0463 | ||
|  | 4b07c38782 | ||
|  | e0256adf77 | ||
|  | 5748f017dd | ||
|  | 973f4c2c2d | ||
|  | 8e1774c69f | ||
|  | 56a36072f7 | ||
|  | 8755d108ed | ||
|  | ea40cd73d1 | ||
|  | 0e28899b72 | ||
|  | eee30db981 | 
| @@ -1,41 +0,0 @@ | ||||
| version: '{branch}.{build}' | ||||
| clone_depth: 1 | ||||
| skip_tags: true | ||||
|  | ||||
| environment: | ||||
|   MSYSTEM: MINGW32 | ||||
|  | ||||
| init: | ||||
|   - git config --global core.autocrlf input | ||||
|  | ||||
| install: | ||||
|   - set PATH=c:\msys64\mingw32\bin;c:\msys64\usr\bin;c:\msys64\bin;%PATH% | ||||
|   - echo %PATH% | ||||
|   - pacman -S --noconfirm --needed make ninja mingw-w64-i686-libusb mingw-w64-i686-sqlite3 mingw-w64-i686-zlib mingw-w64-i686-gcc zip | ||||
|  | ||||
| build_script: | ||||
|   - make | ||||
|   - zip -9 fluxengine.zip fluxengine.exe brother120tool.exe | ||||
|  | ||||
| artifacts: | ||||
|   - path: fluxengine.zip | ||||
|     name: fluxengine.zip | ||||
|  | ||||
| deploy: | ||||
|   release: FluxEngine Windows client version $(APPVEYOR_BUILD_NUMBER) | ||||
|   description: > | ||||
|     This is an automatically built version of the FluxEngine Windows client | ||||
|     which is generated whenever a significant checkin has happened. It's had | ||||
|     no testing whatsoever. | ||||
|  | ||||
|     To use, download it, put it somewhere, and then run it from a cmd window | ||||
|     or other command line shell. | ||||
|   provider: GitHub | ||||
|   auth_token: | ||||
|     secure: dfJjj7fWCoDUz+Ni11OcNPB/U3TNJFwNA2AsL++ChFjniUsZLlC6SDWHiL/t4FZo | ||||
|   artifact: fluxengine.zip | ||||
|   draft: false | ||||
|   prerelease: false | ||||
|   on: | ||||
|     branch: master | ||||
|  | ||||
							
								
								
									
										55
									
								
								.github/workflows/ccpp.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								.github/workflows/ccpp.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,55 @@ | ||||
| name: C/C++ CI | ||||
|  | ||||
| on: [push] | ||||
|  | ||||
| jobs: | ||||
|   build-linux: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v1 | ||||
|       with: | ||||
|         fetch-depth: 1 | ||||
|     - name: apt | ||||
|       run: sudo apt update && sudo apt install libudev-dev libsqlite3-dev ninja-build protobuf-compiler libgtk-3-dev | ||||
|     - name: make | ||||
|       run: make | ||||
|  | ||||
|   build-macos: | ||||
|     runs-on: macos-latest | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|       with: | ||||
|         fetch-depth: 1 | ||||
|     - name: brew | ||||
|       run: brew install sqlite pkg-config libusb ninja protobuf truncate | ||||
|     - name: make | ||||
|       run: make | ||||
|  | ||||
|   build-windows: | ||||
|     runs-on: windows-latest | ||||
|     defaults: | ||||
|       run: | ||||
|         shell: msys2 {0} | ||||
|     steps: | ||||
|     - uses: msys2/setup-msys2@v2 | ||||
|       with: | ||||
|         update: true | ||||
|         msystem: MINGW32 | ||||
|         install: >- | ||||
|           make | ||||
|           ninja | ||||
|           mingw-w64-i686-libusb | ||||
|           mingw-w64-i686-sqlite3 | ||||
|           mingw-w64-i686-zlib | ||||
|           mingw-w64-i686-gcc | ||||
|           zip | ||||
|           mingw-w64-i686-protobuf | ||||
|           vim | ||||
|           diffutils | ||||
|     - uses: actions/checkout@v1 | ||||
|       with: | ||||
|         fetch-depth: 1 | ||||
|     - name: build | ||||
|       run: | | ||||
|         make | ||||
|  | ||||
							
								
								
									
										64
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								.github/workflows/release.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,64 @@ | ||||
| name: Autorelease | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|       - "master" | ||||
|  | ||||
| jobs: | ||||
|   dev-release: | ||||
|     runs-on: windows-latest | ||||
|     defaults: | ||||
|       run: | ||||
|         shell: msys2 {0} | ||||
|     steps: | ||||
|     - uses: msys2/setup-msys2@v2 | ||||
|       with: | ||||
|         update: true | ||||
|         msystem: MINGW32 | ||||
|         install: >- | ||||
|           make | ||||
|           ninja | ||||
|           mingw-w64-i686-libusb | ||||
|           mingw-w64-i686-sqlite3 | ||||
|           mingw-w64-i686-zlib | ||||
|           mingw-w64-i686-gcc | ||||
|           zip | ||||
|           mingw-w64-i686-protobuf | ||||
|           vim | ||||
|           diffutils | ||||
|     - uses: actions/checkout@v2 | ||||
|       with: | ||||
|         fetch-depth: 1 | ||||
|     - name: build | ||||
|       run: | | ||||
|         make | ||||
|     - name: zip | ||||
|       run: | | ||||
|         zip -9 fluxengine.zip fluxengine.exe brother120tool.exe brother240tool.exe FluxEngine.cydsn/CortexM3/ARM_GCC_541/Release/FluxEngine.hex | ||||
|     - name: date | ||||
|       run: | | ||||
|         echo "RELEASE_DATE=$(date --rfc-3339=date)" >> ${GITHUB_ENV} | ||||
|     - name: tag | ||||
|       uses: EndBug/latest-tag@latest | ||||
|       with: | ||||
|         tag-name: dev | ||||
|       env: | ||||
|         GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
|     - name: delete-old-assets | ||||
|       uses: mknejp/delete-release-assets@v1 | ||||
|       with: | ||||
|         token: ${{ github.token }} | ||||
|         tag: dev | ||||
|         assets: |  | ||||
|           fluxengine.zip | ||||
|         fail-if-no-assets: false | ||||
|     - name: release | ||||
|       uses: softprops/action-gh-release@v1 | ||||
|       with: | ||||
|         name: Development build ${{ env.RELEASE_DATE }} | ||||
|         files: | | ||||
|           fluxengine.zip | ||||
|         tag_name: dev | ||||
|       env: | ||||
|         GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }} | ||||
| @@ -3,6 +3,10 @@ streams | ||||
| .*\.flux | ||||
| .*\.img | ||||
| .*\.raw | ||||
| .*\.orig | ||||
| .vscode | ||||
| remote | ||||
| FluxEngine.cydsn/CortexM3 | ||||
| FluxEngine.cydsn/Generated_Source | ||||
| FluxEngine.cydsn/codegentemp | ||||
|  | ||||
|   | ||||
							
								
								
									
										39
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										39
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,39 +0,0 @@ | ||||
| language: shell | ||||
| git: | ||||
|     depth: 1 | ||||
|  | ||||
| matrix: | ||||
|     include: | ||||
|         - | ||||
|             os: linux | ||||
|             sudo: false | ||||
|             dist: xenial | ||||
|             compiler: gcc | ||||
|             env: CXX=g++-8 | ||||
|             script: | ||||
|             - make | ||||
|         - | ||||
|             os: osx | ||||
|             osx_image: xcode10.2 | ||||
|             compiler: clang | ||||
|             env: | ||||
|             - HOMEBREW_NO_INSTALL_CLEANUP=1 | ||||
|  | ||||
| addons: | ||||
|     apt: | ||||
|         sources: | ||||
|         - llvm-toolchain-precise-3.8 | ||||
|         - ubuntu-toolchain-r-test | ||||
|         packages: | ||||
|         - ninja-build | ||||
|         - libusb-1.0-0-dev | ||||
|         - libsqlite3-dev | ||||
|         - g++-8 | ||||
|     homebrew: | ||||
|         packages: | ||||
|         - ninja | ||||
|  | ||||
| script: | ||||
| - make | ||||
|  | ||||
|  | ||||
							
								
								
									
										4626
									
								
								FluxEngine.cydsn/CortexM3/ARM_GCC_541/Release/FluxEngine.hex
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4626
									
								
								FluxEngine.cydsn/CortexM3/ARM_GCC_541/Release/FluxEngine.hex
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										27
									
								
								FluxEngine.cydsn/FIFOin/API/c.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								FluxEngine.cydsn/FIFOin/API/c.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| #include "cyfitter_cfg.h" | ||||
| #include "cydevice_trm.h" | ||||
| #include "cyfitter.h" | ||||
| #include "`$INSTANCE_NAME`_h.h" | ||||
|  | ||||
| void `$INSTANCE_NAME`_Start() | ||||
| { | ||||
|    `$INSTANCE_NAME`_Init(); | ||||
| }     | ||||
|  | ||||
| void `$INSTANCE_NAME`_Stop() | ||||
| { | ||||
|     `$INSTANCE_NAME`_Disable(); | ||||
| } | ||||
|  | ||||
| void `$INSTANCE_NAME`_Init() | ||||
| {     | ||||
|     `$INSTANCE_NAME`_Enable(); | ||||
|      | ||||
| } | ||||
| void `$INSTANCE_NAME`_Enable() | ||||
| { | ||||
| } | ||||
|  | ||||
| void `$INSTANCE_NAME`_Disable() | ||||
| { | ||||
| } | ||||
							
								
								
									
										50
									
								
								FluxEngine.cydsn/FIFOin/API/h.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								FluxEngine.cydsn/FIFOin/API/h.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | ||||
| #if !defined(`$INSTANCE_NAME`_H) | ||||
| #define `$INSTANCE_NAME`_H | ||||
|  | ||||
| #include "cytypes.h" | ||||
| #include "cyfitter.h" | ||||
| #include "CyLib.h"  | ||||
|  | ||||
| #define `$INSTANCE_NAME`_FIFO_PTR	         ((reg8 *) `$INSTANCE_NAME`_dp__F0_REG) | ||||
|  | ||||
|     /* Macros to clear DP FIFOs.*/ | ||||
| #define `$INSTANCE_NAME`_CLEAR do { \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0x01u | \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG)));\ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0xfeu & \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG)));\ | ||||
|     } while(0) | ||||
|  | ||||
| /* Macros to set FIFO level mode. See the TRM for details */ | ||||
| #define `$INSTANCE_NAME`_SET_LEVEL_NORMAL \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0xfbu & \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG))) | ||||
| #define `$INSTANCE_NAME`_SET_LEVEL_MID \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0x04u | \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG))) | ||||
|  | ||||
| /* Macros to set FIFO to single-buffer mode. */ | ||||
| #define `$INSTANCE_NAME`_SINGLE_BUFFER_SET \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0x01u | \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG))) | ||||
|  | ||||
| /* Macros to return the FIFO to normal mode. */ | ||||
| #define `$INSTANCE_NAME`_SINGLE_BUFFER_UNSET \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0xfeu & \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG))) | ||||
|      | ||||
| void `$INSTANCE_NAME`_Enable(); | ||||
| void `$INSTANCE_NAME`_Disable(); | ||||
| void `$INSTANCE_NAME`_Start(); | ||||
| void `$INSTANCE_NAME`_Stop(); | ||||
| void `$INSTANCE_NAME`_Init(); | ||||
|  | ||||
| #endif | ||||
|  | ||||
| /* [] END OF FILE */ | ||||
							
								
								
									
										
											BIN
										
									
								
								FluxEngine.cydsn/FIFOin/FIFOin.cysym
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								FluxEngine.cydsn/FIFOin/FIFOin.cysym
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										128
									
								
								FluxEngine.cydsn/FIFOin/FIFOin.v
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								FluxEngine.cydsn/FIFOin/FIFOin.v
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | ||||
|  | ||||
| //`#start header` -- edit after this line, do not edit this line | ||||
| `include "cypress.v" | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
|  | ||||
| /* Ultra-simple FIFO in component: a byte is shifted in every clock when req | ||||
|  * is high. */ | ||||
|   | ||||
| module FIFOin (drq, clk, d, req); | ||||
| 	output  drq; | ||||
| 	input   clk; | ||||
| 	input  [7:0] d; | ||||
| 	input  req; | ||||
|  | ||||
| //`#start body` -- edit after this line, do not edit this line | ||||
|  | ||||
| wire [7:0] pi; | ||||
| assign pi = d; | ||||
|  | ||||
| wire load; | ||||
| assign load = req; | ||||
|  | ||||
| cy_psoc3_dp #(.cy_dpconfig( | ||||
| { | ||||
| 	`CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
| 	`CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
| 	`CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
| 	`CS_CMP_SEL_CFGA, /*CFGRAM0:    */ | ||||
| 	`CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
| 	`CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
| 	`CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
| 	`CS_CMP_SEL_CFGA, /*CFGRAM1:     */ | ||||
| 	`CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
| 	`CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
| 	`CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
| 	`CS_CMP_SEL_CFGA, /*CFGRAM2:     */ | ||||
| 	`CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
| 	`CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
| 	`CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
| 	`CS_CMP_SEL_CFGA, /*CFGRAM3:     */ | ||||
| 	`CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
| 	`CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
| 	`CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
| 	`CS_CMP_SEL_CFGA, /*CFGRAM4:     */ | ||||
| 	`CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
| 	`CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
| 	`CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
| 	`CS_CMP_SEL_CFGA, /*CFGRAM5:     */ | ||||
| 	`CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
| 	`CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
| 	`CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
| 	`CS_CMP_SEL_CFGA, /*CFGRAM6:     */ | ||||
| 	`CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
| 	`CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
| 	`CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
| 	`CS_CMP_SEL_CFGA, /*CFGRAM7:     */ | ||||
| 	8'hFF, 8'h00,	/*CFG9:     */ | ||||
| 	8'hFF, 8'hFF,	/*CFG11-10:     */ | ||||
| 	`SC_CMPB_A1_D1, `SC_CMPA_A1_D1, `SC_CI_B_ARITH, | ||||
| 	`SC_CI_A_ARITH, `SC_C1_MASK_DSBL, `SC_C0_MASK_DSBL, | ||||
| 	`SC_A_MASK_DSBL, `SC_DEF_SI_0, `SC_SI_B_DEFSI, | ||||
| 	`SC_SI_A_DEFSI, /*CFG13-12:     */ | ||||
| 	`SC_A0_SRC_PIN, `SC_SHIFT_SL, 1'h0, | ||||
| 	1'h0, `SC_FIFO1_BUS, `SC_FIFO0_ALU, | ||||
| 	`SC_MSB_DSBL, `SC_MSB_BIT0, `SC_MSB_NOCHN, | ||||
| 	`SC_FB_NOCHN, `SC_CMP1_NOCHN, | ||||
| 	`SC_CMP0_NOCHN, /*CFG15-14:     */ | ||||
| 	10'h00, `SC_FIFO_CLK__DP,`SC_FIFO_CAP_AX, | ||||
| 	`SC_FIFO_LEVEL,`SC_FIFO__SYNC,`SC_EXTCRC_DSBL, | ||||
| 	`SC_WRK16CAT_DSBL /*CFG17-16:     */ | ||||
| } | ||||
| )) dp( | ||||
| 	/* input          */ .clk(clk), | ||||
| 	/* input [02:00]  */ .cs_addr(3'b0),    // Program counter | ||||
| 	/* input          */ .route_si(1'b0),   // Shift in | ||||
| 	/* input          */ .route_ci(1'b0),   // Carry in | ||||
| 	/* input          */ .f0_load(load),    // Load FIFO 0 | ||||
| 	/* input          */ .f1_load(1'b0), 	// Load FIFO 1 | ||||
| 	/* input          */ .d0_load(1'b0), 	// Load Data Register 0 | ||||
| 	/* input          */ .d1_load(1'b0), 	// Load Data Register 1 | ||||
| 	/* output         */ .ce0(), 			// Accumulator 0 = Data register 0 | ||||
| 	/* output         */ .cl0(), 			// Accumulator 0 < Data register 0 | ||||
| 	/* output         */ .z0(), 			// Accumulator 0 = 0 | ||||
| 	/* output         */ .ff0(), 			// Accumulator 0 = FF | ||||
| 	/* output         */ .ce1(), 			// Accumulator [0|1] = Data register 1 | ||||
| 	/* output         */ .cl1(), 			// Accumulator [0|1] < Data register 1 | ||||
| 	/* output         */ .z1(), 			// Accumulator 1 = 0 | ||||
| 	/* output         */ .ff1(), 			// Accumulator 1 = FF | ||||
| 	/* output         */ .ov_msb(), 		// Operation over flow | ||||
| 	/* output         */ .co_msb(), 		// Carry out | ||||
| 	/* output         */ .cmsb(), 			// Carry out | ||||
| 	/* output         */ .so(), 			// Shift out | ||||
|     /* output         */ .f0_bus_stat(drq), // not empty | ||||
| 	/* output         */ .f0_blk_stat(full),// full | ||||
| 	/* output         */ .f1_bus_stat(), 	// FIFO 1 status to uP | ||||
| 	/* output         */ .f1_blk_stat(), 	// FIFO 1 status to DP | ||||
| 	/* input          */ .ci(1'b0), 		// Carry in from previous stage | ||||
| 	/* output         */ .co(), 			// Carry out to next stage | ||||
| 	/* input          */ .sir(1'b0), 		// Shift in from right side | ||||
| 	/* output         */ .sor(), 			// Shift out to right side | ||||
| 	/* input          */ .sil(1'b0), 		// Shift in from left side | ||||
| 	/* output         */ .sol(), 			// Shift out to left side | ||||
| 	/* input          */ .msbi(1'b0), 		// MSB chain in | ||||
| 	/* output         */ .msbo(), 			// MSB chain out | ||||
| 	/* input [01:00]  */ .cei(2'b0),        // Compare equal in from prev stage | ||||
| 	/* output [01:00] */ .ceo(),            // Compare equal out to next stage | ||||
| 	/* input [01:00]  */ .cli(2'b0), 	    // Compare less than in from prv stage | ||||
| 	/* output [01:00] */ .clo(),            // Compare less than out to next stage | ||||
| 	/* input [01:00]  */ .zi(2'b0),         // Zero detect in from previous stage | ||||
| 	/* output [01:00] */ .zo(),             // Zero detect out to next stage | ||||
| 	/* input [01:00]  */ .fi(2'b0), 		// 0xFF detect in from previous stage | ||||
| 	/* output [01:00] */ .fo(), 	        // 0xFF detect out to next stage | ||||
| 	/* input [01:00]  */ .capi(2'b0),	    // Capture in from previous stage | ||||
| 	/* output [01:00] */ .capo(),		    // Capture out to next stage | ||||
| 	/* input          */ .cfbi(1'b0), 		// CRC Feedback in from previous stage | ||||
| 	/* output         */ .cfbo(), 			// CRC Feedback out to next stage | ||||
| 	/* input [07:00]  */ .pi(pi), 		    // Parallel data port | ||||
| 	/* output [07:00] */ .po()              // Parallel data port | ||||
| ); | ||||
|  | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
| endmodule | ||||
| //`#start footer` -- edit after this line, do not edit this line | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
							
								
								
									
										27
									
								
								FluxEngine.cydsn/FIFOout/API/c.c
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								FluxEngine.cydsn/FIFOout/API/c.c
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| #include "cyfitter_cfg.h" | ||||
| #include "cydevice_trm.h" | ||||
| #include "cyfitter.h" | ||||
| #include "`$INSTANCE_NAME`_h.h" | ||||
|  | ||||
| void `$INSTANCE_NAME`_Start() | ||||
| { | ||||
|    `$INSTANCE_NAME`_Init(); | ||||
| }     | ||||
|  | ||||
| void `$INSTANCE_NAME`_Stop() | ||||
| { | ||||
|     `$INSTANCE_NAME`_Disable(); | ||||
| } | ||||
|  | ||||
| void `$INSTANCE_NAME`_Init() | ||||
| {     | ||||
|     `$INSTANCE_NAME`_Enable(); | ||||
|      | ||||
| } | ||||
| void `$INSTANCE_NAME`_Enable() | ||||
| { | ||||
| } | ||||
|  | ||||
| void `$INSTANCE_NAME`_Disable() | ||||
| { | ||||
| } | ||||
							
								
								
									
										50
									
								
								FluxEngine.cydsn/FIFOout/API/h.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										50
									
								
								FluxEngine.cydsn/FIFOout/API/h.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,50 @@ | ||||
| #if !defined(`$INSTANCE_NAME`_H) | ||||
| #define `$INSTANCE_NAME`_H | ||||
|  | ||||
| #include "cytypes.h" | ||||
| #include "cyfitter.h" | ||||
| #include "CyLib.h"  | ||||
|  | ||||
| #define `$INSTANCE_NAME`_FIFO_PTR	         ((reg8 *) `$INSTANCE_NAME`_dp__F0_REG) | ||||
|  | ||||
|     /* Macros to clear DP FIFOs.*/ | ||||
| #define `$INSTANCE_NAME`_CLEAR do { \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0x01u | \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG)));\ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0xfeu & \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG)));\ | ||||
|     } while(0) | ||||
|  | ||||
| /* Macros to set FIFO level mode. See the TRM for details */ | ||||
| #define `$INSTANCE_NAME`_SET_LEVEL_NORMAL \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0xfbu & \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG))) | ||||
| #define `$INSTANCE_NAME`_SET_LEVEL_MID \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0x04u | \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG))) | ||||
|  | ||||
| /* Macros to set FIFO to single-buffer mode. */ | ||||
| #define `$INSTANCE_NAME`_SINGLE_BUFFER_SET \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0x01u | \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG))) | ||||
|  | ||||
| /* Macros to return the FIFO to normal mode. */ | ||||
| #define `$INSTANCE_NAME`_SINGLE_BUFFER_UNSET \ | ||||
|     CY_SET_XTND_REG8(\ | ||||
|         ((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG), 0xfeu & \ | ||||
|         CY_GET_XTND_REG8(((reg8 *) `$INSTANCE_NAME`_dp__DP_AUX_CTL_REG))) | ||||
|      | ||||
| void `$INSTANCE_NAME`_Enable(); | ||||
| void `$INSTANCE_NAME`_Disable(); | ||||
| void `$INSTANCE_NAME`_Start(); | ||||
| void `$INSTANCE_NAME`_Stop(); | ||||
| void `$INSTANCE_NAME`_Init(); | ||||
|  | ||||
| #endif | ||||
|  | ||||
| /* [] END OF FILE */ | ||||
							
								
								
									
										
											BIN
										
									
								
								FluxEngine.cydsn/FIFOout/FIFOout.cysym
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								FluxEngine.cydsn/FIFOout/FIFOout.cysym
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										169
									
								
								FluxEngine.cydsn/FIFOout/FIFOout.v
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										169
									
								
								FluxEngine.cydsn/FIFOout/FIFOout.v
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,169 @@ | ||||
|  | ||||
| //`#start header` -- edit after this line, do not edit this line | ||||
| `include "cypress.v" | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
| // Generated on 11/16/2017 at 15:44 | ||||
| // Component: FIFOout | ||||
| module FIFOout ( | ||||
|     input req, | ||||
| 	input clk, | ||||
|     output [7:0] d, | ||||
| 	output drq, | ||||
| 	output empty, | ||||
|     output ack | ||||
| ); | ||||
|  | ||||
| //`#start body` -- edit after this line, do not edit this line | ||||
|  | ||||
| /* Reads from the FIFO are done based on the FIFO being not empty. */ | ||||
|   | ||||
| wire [7:0] po; | ||||
| assign d = po; | ||||
|  | ||||
| localparam STATE_WAITFORREQ = 0; | ||||
| localparam STATE_READFROMFIFO = 1; | ||||
| localparam STATE_WAITFORNREQ = 2; | ||||
|  | ||||
| reg [1:0] state; | ||||
| wire readfromfifo; | ||||
|  | ||||
| assign ack = (state == STATE_WAITFORNREQ); | ||||
| assign readfromfifo = (state == STATE_READFROMFIFO); | ||||
|  | ||||
| always @(posedge clk) | ||||
| begin | ||||
|     case (state) | ||||
|         /* opcode is not valid; req is low; wait for req to go high. */ | ||||
|         STATE_WAITFORREQ: | ||||
|         begin | ||||
|             if (!empty && req) | ||||
|                 state <= STATE_READFROMFIFO; | ||||
|         end | ||||
|          | ||||
|         /* Fetch a single value from the FIFO. */ | ||||
|         STATE_READFROMFIFO: | ||||
|             state <= STATE_WAITFORNREQ; | ||||
|              | ||||
|         /* opcode is valid; ack is high. Wait for req to go low. */ | ||||
|         STATE_WAITFORNREQ: | ||||
|             if (!req) | ||||
|                 state <= STATE_WAITFORREQ; | ||||
|     endcase | ||||
| end | ||||
|              | ||||
| cy_psoc3_dp #(.cy_dpconfig( | ||||
| { | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM0: idle */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC___F0, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM1: read from fifo */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM2:           */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM3:           */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM4:           */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM5:           */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM6:           */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM7:           */ | ||||
|     8'hFF, 8'h00,  /*CFG9:           */ | ||||
|     8'hFF, 8'hFF,  /*CFG11-10:           */ | ||||
|     `SC_CMPB_A1_D1, `SC_CMPA_A1_D1, `SC_CI_B_ARITH, | ||||
|     `SC_CI_A_ARITH, `SC_C1_MASK_DSBL, `SC_C0_MASK_DSBL, | ||||
|     `SC_A_MASK_DSBL, `SC_DEF_SI_0, `SC_SI_B_DEFSI, | ||||
|     `SC_SI_A_DEFSI, /*CFG13-12:           */ | ||||
|     `SC_A0_SRC_ACC, `SC_SHIFT_SL, 1'h0, | ||||
|     1'h0, `SC_FIFO1_BUS, `SC_FIFO0_BUS, | ||||
|     `SC_MSB_DSBL, `SC_MSB_BIT0, `SC_MSB_NOCHN, | ||||
|     `SC_FB_NOCHN, `SC_CMP1_NOCHN, | ||||
|     `SC_CMP0_NOCHN, /*CFG15-14:           */ | ||||
|     10'h00, `SC_FIFO_CLK__DP,`SC_FIFO_CAP_AX, | ||||
|     `SC_FIFO_LEVEL,`SC_FIFO_ASYNC,`SC_EXTCRC_DSBL, | ||||
|     `SC_WRK16CAT_DSBL /*CFG17-16:           */ | ||||
| } | ||||
| )) dp( | ||||
|         /*  input                   */  .reset(1'b0), | ||||
|         /*  input                   */  .clk(clk), | ||||
|         /*  input   [02:00]         */  .cs_addr({2'b0, readfromfifo}), | ||||
|         /*  input                   */  .route_si(1'b0), | ||||
|         /*  input                   */  .route_ci(1'b0), | ||||
|         /*  input                   */  .f0_load(1'b0), | ||||
|         /*  input                   */  .f1_load(1'b0), | ||||
|         /*  input                   */  .d0_load(1'b0), | ||||
|         /*  input                   */  .d1_load(1'b0), | ||||
|         /*  output                  */  .ce0(), | ||||
|         /*  output                  */  .cl0(), | ||||
|         /*  output                  */  .z0(), | ||||
|         /*  output                  */  .ff0(), | ||||
|         /*  output                  */  .ce1(), | ||||
|         /*  output                  */  .cl1(), | ||||
|         /*  output                  */  .z1(), | ||||
|         /*  output                  */  .ff1(), | ||||
|         /*  output                  */  .ov_msb(), | ||||
|         /*  output                  */  .co_msb(), | ||||
|         /*  output                  */  .cmsb(), | ||||
|         /*  output                  */  .so(), | ||||
|         /*  output                  */  .f0_bus_stat(drq), // not full | ||||
|         /*  output                  */  .f0_blk_stat(empty), // empty | ||||
|         /*  output                  */  .f1_bus_stat(), | ||||
|         /*  output                  */  .f1_blk_stat(), | ||||
|          | ||||
|         /* input                    */  .ci(1'b0),     // Carry in from previous stage | ||||
|         /* output                   */  .co(),// Carry out to next stage | ||||
|         /* input                    */  .sir(1'b0),    // Shift in from right side | ||||
|         /* output                   */  .sor(),        // Shift out to right side | ||||
|         /* input                    */  .sil(1'b0),    // Shift in from left side | ||||
|         /* output                   */  .sol(),        // Shift out to left side | ||||
|         /* input                    */  .msbi(1'b0),   // MSB chain in | ||||
|         /* output                   */  .msbo(),       // MSB chain out | ||||
|         /* input [01:00]            */  .cei(2'b0),    // Compare equal in from prev stage | ||||
|         /* output [01:00]           */  .ceo(),        // Compare equal out to next stage | ||||
|         /* input [01:00]            */  .cli(2'b0),    // Compare less than in from prv stage | ||||
|         /* output [01:00]           */  .clo(),        // Compare less than out to next stage | ||||
|         /* input [01:00]            */  .zi(2'b0),     // Zero detect in from previous stage | ||||
|         /* output [01:00]           */  .zo(),         // Zero detect out to next stage | ||||
|         /* input [01:00]            */  .fi(2'b0),     // 0xFF detect in from previous stage | ||||
|         /* output [01:00]           */  .fo(),         // 0xFF detect out to next stage | ||||
|         /* input [01:00]            */  .capi(2'b0),   // Software capture from previous stage | ||||
|         /* output [01:00]           */  .capo(),       // Software capture to next stage | ||||
|         /* input                    */  .cfbi(1'b0),   // CRC Feedback in from previous stage | ||||
|         /* output                   */  .cfbo(),       // CRC Feedback out to next stage | ||||
|         /* input [07:00]            */  .pi(8'b0),     // Parallel data port | ||||
|         /* output [07:00]           */  .po(po)       // Parallel data port | ||||
| ); | ||||
|  | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
| endmodule | ||||
| //`#start footer` -- edit after this line, do not edit this line | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
| @@ -28,6 +28,54 @@ | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="1a7e8637-3b6b-4e84-839c-0dfc18fdaf5b"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
|         <Data key="derive_type" value="NAMED_DIVIDER" /> | ||||
|         <Data key="desired_freq" value="0" /> | ||||
|         <Data key="desired_unit" value="15" /> | ||||
|         <Data key="divider" value="0" /> | ||||
|         <Data key="domain" value="DIGITAL" /> | ||||
|         <Data key="enabled" value="True" /> | ||||
|         <Data key="minus_accuracy" value="0.25" /> | ||||
|         <Data key="minus_tolerance" value="5" /> | ||||
|         <Data key="name" value="Clock_5" /> | ||||
|         <Data key="named_src_direct_connect" value="True" /> | ||||
|         <Data key="netlist_name" value="Clock_5" /> | ||||
|         <Data key="placement" value="AUTO" /> | ||||
|         <Data key="plus_accuracy" value="0.25" /> | ||||
|         <Data key="plus_tolerance" value="5" /> | ||||
|         <Data key="scope" value="LOCAL" /> | ||||
|         <Data key="src_clk_id" value="75C2148C-3656-4d8a-846D-0CAE99AB6FF7" /> | ||||
|         <Data key="src_clk_name" value="BUS_CLK" /> | ||||
|         <Data key="start_on_reset" value="True" /> | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="3f3708ae-fb62-4012-919b-9a3b9a1dfbc2"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
|         <Data key="derive_type" value="NAMED_DIVIDER" /> | ||||
|         <Data key="desired_freq" value="0" /> | ||||
|         <Data key="desired_unit" value="15" /> | ||||
|         <Data key="divider" value="0" /> | ||||
|         <Data key="domain" value="DIGITAL" /> | ||||
|         <Data key="enabled" value="True" /> | ||||
|         <Data key="minus_accuracy" value="0.25" /> | ||||
|         <Data key="minus_tolerance" value="5" /> | ||||
|         <Data key="name" value="Clock_8" /> | ||||
|         <Data key="named_src_direct_connect" value="True" /> | ||||
|         <Data key="netlist_name" value="Clock_8" /> | ||||
|         <Data key="placement" value="AUTO" /> | ||||
|         <Data key="plus_accuracy" value="0.25" /> | ||||
|         <Data key="plus_tolerance" value="5" /> | ||||
|         <Data key="scope" value="LOCAL" /> | ||||
|         <Data key="src_clk_id" value="75C2148C-3656-4d8a-846D-0CAE99AB6FF7" /> | ||||
|         <Data key="src_clk_name" value="BUS_CLK" /> | ||||
|         <Data key="start_on_reset" value="True" /> | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="4eef02b9-8ad1-43c4-85f1-b3335faa5fc4"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
| @@ -147,6 +195,54 @@ | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="71bc291d-84a7-40a8-b7b2-1c8a34326a31"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
|         <Data key="derive_type" value="NAMED_FREQ" /> | ||||
|         <Data key="desired_freq" value="300" /> | ||||
|         <Data key="desired_unit" value="0" /> | ||||
|         <Data key="divider" value="65536" /> | ||||
|         <Data key="domain" value="DIGITAL" /> | ||||
|         <Data key="enabled" value="True" /> | ||||
|         <Data key="minus_accuracy" value="0.25" /> | ||||
|         <Data key="minus_tolerance" value="5" /> | ||||
|         <Data key="name" value="CLOCK300" /> | ||||
|         <Data key="named_src_direct_connect" value="False" /> | ||||
|         <Data key="netlist_name" value="CLOCK300" /> | ||||
|         <Data key="placement" value="AUTO" /> | ||||
|         <Data key="plus_accuracy" value="0.25" /> | ||||
|         <Data key="plus_tolerance" value="5" /> | ||||
|         <Data key="scope" value="LOCAL" /> | ||||
|         <Data key="src_clk_id" value="CEF43CFB-0213-49b9-B980-2FFAB81C5B47" /> | ||||
|         <Data key="src_clk_name" value="IMO" /> | ||||
|         <Data key="start_on_reset" value="True" /> | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="90ce0c72-9f10-44ef-a049-f0f525d59bea"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
|         <Data key="derive_type" value="NAMED_FREQ" /> | ||||
|         <Data key="desired_freq" value="128" /> | ||||
|         <Data key="desired_unit" value="0" /> | ||||
|         <Data key="divider" value="65536" /> | ||||
|         <Data key="domain" value="DIGITAL" /> | ||||
|         <Data key="enabled" value="True" /> | ||||
|         <Data key="minus_accuracy" value="0.25" /> | ||||
|         <Data key="minus_tolerance" value="5" /> | ||||
|         <Data key="name" value="CLOCK8" /> | ||||
|         <Data key="named_src_direct_connect" value="False" /> | ||||
|         <Data key="netlist_name" value="CLOCK8" /> | ||||
|         <Data key="placement" value="AUTO" /> | ||||
|         <Data key="plus_accuracy" value="0.25" /> | ||||
|         <Data key="plus_tolerance" value="5" /> | ||||
|         <Data key="scope" value="LOCAL" /> | ||||
|         <Data key="src_clk_id" value="CEF43CFB-0213-49b9-B980-2FFAB81C5B47" /> | ||||
|         <Data key="src_clk_name" value="IMO" /> | ||||
|         <Data key="start_on_reset" value="True" /> | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="349ffa20-8576-4ac3-9a6f-34ef606de6cf"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
| @@ -170,6 +266,29 @@ | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="4033c29d-f4bc-4e94-ac95-aa587e869f88/696a0979-21fc-4185-bf38-6c79febcde7a"> | ||||
|         <Data key="check_tolerance" value="False" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
|         <Data key="derive_type" value="AUTO" /> | ||||
|         <Data key="desired_freq" value="1600000" /> | ||||
|         <Data key="desired_unit" value="0" /> | ||||
|         <Data key="divider" value="40" /> | ||||
|         <Data key="domain" value="DIGITAL" /> | ||||
|         <Data key="enabled" value="True" /> | ||||
|         <Data key="minus_accuracy" value="0.25" /> | ||||
|         <Data key="minus_tolerance" value="5" /> | ||||
|         <Data key="name" value="OUTPUT_VOLTAGE_ADC_theACLK" /> | ||||
|         <Data key="netlist_name" value="\OUTPUT_VOLTAGE_ADC:theACLK\" /> | ||||
|         <Data key="placement" value="AUTO" /> | ||||
|         <Data key="plus_accuracy" value="0.25" /> | ||||
|         <Data key="plus_tolerance" value="5" /> | ||||
|         <Data key="scope" value="LOCAL" /> | ||||
|         <Data key="src_clk_id" value="61737EF6-3B74-48f9-8B91-F7473A442AE7" /> | ||||
|         <Data key="src_clk_name" value="MASTER_CLK" /> | ||||
|         <Data key="start_on_reset" value="True" /> | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="6616e828-6611-4893-a674-66c861d79d6c"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
| @@ -241,6 +360,53 @@ | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="09974428-e912-491f-8d2f-361ba50e7599"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
|         <Data key="derive_type" value="NAMED_DIVIDER" /> | ||||
|         <Data key="desired_freq" value="0" /> | ||||
|         <Data key="desired_unit" value="15" /> | ||||
|         <Data key="divider" value="0" /> | ||||
|         <Data key="domain" value="DIGITAL" /> | ||||
|         <Data key="enabled" value="True" /> | ||||
|         <Data key="minus_accuracy" value="0.25" /> | ||||
|         <Data key="minus_tolerance" value="5" /> | ||||
|         <Data key="name" value="Clock_6" /> | ||||
|         <Data key="named_src_direct_connect" value="True" /> | ||||
|         <Data key="netlist_name" value="Clock_6" /> | ||||
|         <Data key="placement" value="AUTO" /> | ||||
|         <Data key="plus_accuracy" value="0.25" /> | ||||
|         <Data key="plus_tolerance" value="5" /> | ||||
|         <Data key="scope" value="LOCAL" /> | ||||
|         <Data key="src_clk_id" value="75C2148C-3656-4d8a-846D-0CAE99AB6FF7" /> | ||||
|         <Data key="src_clk_name" value="BUS_CLK" /> | ||||
|         <Data key="start_on_reset" value="True" /> | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="a5825a94-fa18-4e4f-a843-bc687cacbd56/696a0979-21fc-4185-bf38-6c79febcde7a"> | ||||
|         <Data key="check_tolerance" value="False" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
|         <Data key="derive_type" value="AUTO" /> | ||||
|         <Data key="desired_freq" value="1600000" /> | ||||
|         <Data key="desired_unit" value="0" /> | ||||
|         <Data key="divider" value="40" /> | ||||
|         <Data key="domain" value="DIGITAL" /> | ||||
|         <Data key="enabled" value="True" /> | ||||
|         <Data key="minus_accuracy" value="0.25" /> | ||||
|         <Data key="minus_tolerance" value="5" /> | ||||
|         <Data key="name" value="INPUT_VOLTAGE_ADC_theACLK" /> | ||||
|         <Data key="netlist_name" value="\INPUT_VOLTAGE_ADC:theACLK\" /> | ||||
|         <Data key="placement" value="AUTO" /> | ||||
|         <Data key="plus_accuracy" value="0.25" /> | ||||
|         <Data key="plus_tolerance" value="5" /> | ||||
|         <Data key="scope" value="LOCAL" /> | ||||
|         <Data key="src_clk_id" value="61737EF6-3B74-48f9-8B91-F7473A442AE7" /> | ||||
|         <Data key="src_clk_name" value="MASTER_CLK" /> | ||||
|         <Data key="start_on_reset" value="True" /> | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="b762c287-7f87-4b21-982e-84be01dc5115"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
| @@ -288,6 +454,30 @@ | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="b722443b-8f81-46dc-bf9b-c95eb62bc181"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
|         <Data key="derive_type" value="NAMED_DIVIDER" /> | ||||
|         <Data key="desired_freq" value="0" /> | ||||
|         <Data key="desired_unit" value="15" /> | ||||
|         <Data key="divider" value="0" /> | ||||
|         <Data key="domain" value="DIGITAL" /> | ||||
|         <Data key="enabled" value="True" /> | ||||
|         <Data key="minus_accuracy" value="0.25" /> | ||||
|         <Data key="minus_tolerance" value="5" /> | ||||
|         <Data key="name" value="Clock_1" /> | ||||
|         <Data key="named_src_direct_connect" value="True" /> | ||||
|         <Data key="netlist_name" value="Clock_1" /> | ||||
|         <Data key="placement" value="AUTO" /> | ||||
|         <Data key="plus_accuracy" value="0.25" /> | ||||
|         <Data key="plus_tolerance" value="5" /> | ||||
|         <Data key="scope" value="LOCAL" /> | ||||
|         <Data key="src_clk_id" value="75C2148C-3656-4d8a-846D-0CAE99AB6FF7" /> | ||||
|         <Data key="src_clk_name" value="BUS_CLK" /> | ||||
|         <Data key="start_on_reset" value="True" /> | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="cb7e877c-9fb4-4fc1-a708-f1e48eb5a68c"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
| @@ -312,6 +502,30 @@ | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="d3075dc6-05c8-4dc9-9959-cf7014c0e66f"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
|         <Data key="derive_type" value="NAMED_DIVIDER" /> | ||||
|         <Data key="desired_freq" value="0" /> | ||||
|         <Data key="desired_unit" value="15" /> | ||||
|         <Data key="divider" value="0" /> | ||||
|         <Data key="domain" value="DIGITAL" /> | ||||
|         <Data key="enabled" value="True" /> | ||||
|         <Data key="minus_accuracy" value="0.25" /> | ||||
|         <Data key="minus_tolerance" value="5" /> | ||||
|         <Data key="name" value="Clock_7" /> | ||||
|         <Data key="named_src_direct_connect" value="True" /> | ||||
|         <Data key="netlist_name" value="Clock_7" /> | ||||
|         <Data key="placement" value="AUTO" /> | ||||
|         <Data key="plus_accuracy" value="0.25" /> | ||||
|         <Data key="plus_tolerance" value="5" /> | ||||
|         <Data key="scope" value="LOCAL" /> | ||||
|         <Data key="src_clk_id" value="75C2148C-3656-4d8a-846D-0CAE99AB6FF7" /> | ||||
|         <Data key="src_clk_name" value="BUS_CLK" /> | ||||
|         <Data key="start_on_reset" value="True" /> | ||||
|         <Data key="sync_with_bus_clk" value="True" /> | ||||
|         <Data key="user_set_domain" value="False" /> | ||||
|       </Group> | ||||
|       <Group key="e4a53a4c-40e1-4747-a72a-10193ffdf31c"> | ||||
|         <Data key="check_tolerance" value="True" /> | ||||
|         <Data key="clock_version" value="v1" /> | ||||
| @@ -600,54 +814,69 @@ | ||||
|   </Group> | ||||
|   <Group key="Component"> | ||||
|     <Group key="v1"> | ||||
|       <Data key="cy_boot" value="cy_boot_v5_80" /> | ||||
|       <Data key="cy_boot" value="cy_boot_v6_10" /> | ||||
|       <Data key="Em_EEPROM_Dynamic" value="Em_EEPROM_Dynamic_v2_20" /> | ||||
|       <Data key="LIN_Dynamic" value="LIN_Dynamic_v5_0" /> | ||||
|       <Data key="LIN_Dynamic" value="LIN_Dynamic_v6_0" /> | ||||
|     </Group> | ||||
|   </Group> | ||||
|   <Data key="DataVersionKey" value="2" /> | ||||
|   <Group key="DWRInstGuidMapping"> | ||||
|     <Group key="Clock"> | ||||
|       <Data key="0b2f9bbb-00ce-4115-a788-ffb9d046a9e5" value="Clock_4" /> | ||||
|       <Data key="1a7e8637-3b6b-4e84-839c-0dfc18fdaf5b" value="Clock_5" /> | ||||
|       <Data key="3f3708ae-fb62-4012-919b-9a3b9a1dfbc2" value="Clock_8" /> | ||||
|       <Data key="4eef02b9-8ad1-43c4-85f1-b3335faa5fc4" value="Clock_3" /> | ||||
|       <Data key="06c4d5d4-f15f-4b29-a1d0-c24b2e38b1ec" value="CounterClock" /> | ||||
|       <Data key="24cd38f7-f472-4403-837f-86807c8f5333" value="PULSE_CLOCK" /> | ||||
|       <Data key="63ed4137-0b09-4256-8a27-35c9a2653f1a" value="Clock_2" /> | ||||
|       <Data key="66f14071-bddd-4b4d-a9aa-a129cceaa7b6" value="Clock_3" /> | ||||
|       <Data key="71bc291d-84a7-40a8-b7b2-1c8a34326a31" value="CLOCK300" /> | ||||
|       <Data key="90ce0c72-9f10-44ef-a049-f0f525d59bea" value="CLOCK8" /> | ||||
|       <Data key="349ffa20-8576-4ac3-9a6f-34ef606de6cf" value="Clock_1" /> | ||||
|       <Data key="4033c29d-f4bc-4e94-ac95-aa587e869f88/696a0979-21fc-4185-bf38-6c79febcde7a" value="OUTPUT_VOLTAGE_ADC_theACLK" /> | ||||
|       <Data key="6616e828-6611-4893-a674-66c861d79d6c" value="SignalSamplingClock" /> | ||||
|       <Data key="12664fc6-9d70-44b1-8a49-887a292e1b7f" value="Clock_3" /> | ||||
|       <Data key="75187c05-9501-4450-b306-6ccdd3bb77db" value="Clock_5" /> | ||||
|       <Data key="09974428-e912-491f-8d2f-361ba50e7599" value="Clock_6" /> | ||||
|       <Data key="a5825a94-fa18-4e4f-a843-bc687cacbd56/696a0979-21fc-4185-bf38-6c79febcde7a" value="INPUT_VOLTAGE_ADC_theACLK" /> | ||||
|       <Data key="b762c287-7f87-4b21-982e-84be01dc5115" value="Clock_2" /> | ||||
|       <Data key="b0162966-0060-4af5-82d1-fcb491ad7619/be0a0e37-ad17-42ca-b5a1-1a654d736358" value="UART_IntClock" /> | ||||
|       <Data key="b722443b-8f81-46dc-bf9b-c95eb62bc181" value="Clock_1" /> | ||||
|       <Data key="cb7e877c-9fb4-4fc1-a708-f1e48eb5a68c" value="CounterClock" /> | ||||
|       <Data key="d3075dc6-05c8-4dc9-9959-cf7014c0e66f" value="Clock_7" /> | ||||
|       <Data key="e4a53a4c-40e1-4747-a72a-10193ffdf31c" value="Clock_1" /> | ||||
|       <Data key="efd5f185-0c32-4824-ba72-3ceb5356f5a7" value="Clock_1" /> | ||||
|     </Group> | ||||
|     <Group key="Pin"> | ||||
|       <Data key="3e1862bb-be82-47b0-9549-7ebfe76b6f7b" value="Pin_2" /> | ||||
|       <Data key="4a398466-709f-4228-9500-96178658e13e" value="RDATA" /> | ||||
|       <Data key="5a3407c1-b434-4438-a7b4-b9dfd2280495" value="MOTEA" /> | ||||
|       <Data key="8d318d8b-cf7b-4b6b-b02c-ab1c5c49d0ba" value="SW1" /> | ||||
|       <Data key="8fc20a4f-e4d1-44b3-a5d4-546e8628d61e" value="LED" /> | ||||
|       <Data key="12e00eac-69b5-4717-85c8-25ef6b224d4c" value="DEBUG_PINS" /> | ||||
|       <Data key="41e2d8ed-5494-4d8c-8ff7-f4f789cece51" value="REDWC" /> | ||||
|       <Data key="264be2d3-9481-494b-8d9c-c1905a45e9cc" value="FDD" /> | ||||
|       <Data key="472f8fdb-f772-44fb-8897-cc690694237b" value="WDATA" /> | ||||
|       <Data key="736cb12b-c863-43d4-a8f0-42f06023f8b5" value="SIDE1" /> | ||||
|       <Data key="4249c923-fcff-453b-8629-bec6fddd00c1" value="STEP" /> | ||||
|       <Data key="27315b0e-6a8c-4b7f-be77-73ab434fa803" value="Pin_1" /> | ||||
|       <Data key="1425177d-0d0e-4468-8bcc-e638e5509a9b" value="UartRx" /> | ||||
|       <Data key="a5d987c6-e45b-45b9-ad93-656fab06d720" value="TRK00" /> | ||||
|       <Data key="a93ef5b3-00f4-42c0-8fad-0e275a7e2537" value="MOTEB" /> | ||||
|       <Data key="b8380fb7-fdb8-449f-bd8d-c4ca96cdf55a" value="DEBUG_PINS" /> | ||||
|       <Data key="bc2e8987-db82-469c-bf6f-22fd3464cc70" value="DEBUG_PINS" /> | ||||
|       <Data key="bc5d52a1-1b25-4aa0-9ba9-3f81d122772f" value="DEBUG_PINS" /> | ||||
|       <Data key="beca5e2d-f70f-4900-a4db-7eca1ed3126e/8b77a6c4-10a0-4390-971c-672353e2a49c" value="USBFS_Dm" /> | ||||
|       <Data key="beca5e2d-f70f-4900-a4db-7eca1ed3126e/618a72fc-5ddd-4df5-958f-a3d55102db42" value="USBFS_Dp" /> | ||||
|       <Data key="c5367cde-21d5-4866-9a32-d16abfea0c61" value="WPT" /> | ||||
|       <Data key="d19368c5-6855-41bb-a9ff-808938abef00" value="INDEX" /> | ||||
|       <Data key="e9f14b5a-b2bf-49b8-98f3-d7b5a43ace8d" value="DRVSB" /> | ||||
|       <Data key="e851a3b9-efb8-48be-bbb8-b303b216c393" value="LED" /> | ||||
|       <Data key="e16b5ef8-00d3-40a4-bc1c-194983c8eb3d" value="LOW_CURRENT" /> | ||||
|       <Data key="e851a3b9-efb8-48be-bbb8-b303b216c393" value="INDEX300" /> | ||||
|       <Data key="e51063a9-4fad-40c7-a06b-7cc4b137dc18" value="DSKCHG" /> | ||||
|       <Data key="ea7ee228-8b3f-426c-8bb8-cd7a81937769" value="DIR" /> | ||||
|       <Data key="ed092b9b-d398-4703-be89-cebf998501f6" value="UartTx" /> | ||||
|       <Data key="f9a7371a-8a7d-4144-8b08-69e3d2a3a663" value="INDEX360" /> | ||||
|       <Data key="fbd1f839-40f9-498e-a48b-5f3048ea5c3d/52f31aa9-2f0a-497d-9a1f-1424095e13e6" value="UART_tx" /> | ||||
|       <Data key="fede1767-f3fd-4021-b3d7-8f9d88f36f9b" value="DRVSA" /> | ||||
|       <Data key="fff78075-035e-43d7-8577-bc5be4d21926" value="WGATE" /> | ||||
| @@ -3737,6 +3966,11 @@ | ||||
|     </Group> | ||||
|   </Group> | ||||
|   <Group key="Pin2"> | ||||
|     <Group key="3e1862bb-be82-47b0-9549-7ebfe76b6f7b"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="0,6" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="4a398466-709f-4228-9500-96178658e13e"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="1,5" /> | ||||
| @@ -3752,6 +3986,11 @@ | ||||
|         <Data key="Port Format" value="2,2" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="8fc20a4f-e4d1-44b3-a5d4-546e8628d61e"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="2,1" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="12e00eac-69b5-4717-85c8-25ef6b224d4c"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="2,2" /> | ||||
| @@ -3833,6 +4072,11 @@ | ||||
|         <Data key="Port Format" value="1,0" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="27315b0e-6a8c-4b7f-be77-73ab434fa803"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="0,7" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="1425177d-0d0e-4468-8bcc-e638e5509a9b"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="12,6" /> | ||||
| @@ -3859,6 +4103,32 @@ | ||||
|         <Data key="Port Format" value="2,3" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="bc2e8987-db82-469c-bf6f-22fd3464cc70"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="0,0" /> | ||||
|       </Group> | ||||
|       <Group key="1"> | ||||
|         <Data key="Port Format" value="0,1" /> | ||||
|       </Group> | ||||
|       <Group key="2"> | ||||
|         <Data key="Port Format" value="0,2" /> | ||||
|       </Group> | ||||
|       <Group key="3"> | ||||
|         <Data key="Port Format" value="0,3" /> | ||||
|       </Group> | ||||
|       <Group key="4"> | ||||
|         <Data key="Port Format" value="0,4" /> | ||||
|       </Group> | ||||
|       <Group key="5"> | ||||
|         <Data key="Port Format" value="0,5" /> | ||||
|       </Group> | ||||
|       <Group key="6"> | ||||
|         <Data key="Port Format" value="0,6" /> | ||||
|       </Group> | ||||
|       <Group key="7"> | ||||
|         <Data key="Port Format" value="0,7" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="bc5d52a1-1b25-4aa0-9ba9-3f81d122772f"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="0,5" /> | ||||
| @@ -3892,9 +4162,14 @@ | ||||
|         <Data key="Port Format" value="12,3" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="e16b5ef8-00d3-40a4-bc1c-194983c8eb3d"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="3,2" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="e851a3b9-efb8-48be-bbb8-b303b216c393"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="2,1" /> | ||||
|         <Data key="Port Format" value="3,0" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="e51063a9-4fad-40c7-a06b-7cc4b137dc18"> | ||||
| @@ -3912,9 +4187,14 @@ | ||||
|         <Data key="Port Format" value="12,7" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="f9a7371a-8a7d-4144-8b08-69e3d2a3a663"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="3,1" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="fbd1f839-40f9-498e-a48b-5f3048ea5c3d/52f31aa9-2f0a-497d-9a1f-1424095e13e6"> | ||||
|       <Group key="0"> | ||||
|         <Data key="Port Format" value="2,5" /> | ||||
|         <Data key="Port Format" value="12,7" /> | ||||
|       </Group> | ||||
|     </Group> | ||||
|     <Group key="fede1767-f3fd-4021-b3d7-8f9d88f36f9b"> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										96
									
								
								FluxEngine.cydsn/Sampler/Sampler.v
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										96
									
								
								FluxEngine.cydsn/Sampler/Sampler.v
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,96 @@ | ||||
|  | ||||
| //`#start header` -- edit after this line, do not edit this line | ||||
| `include "cypress.v" | ||||
|  | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
| // Generated on 12/11/2019 at 21:18 | ||||
| // Component: Sampler | ||||
| module Sampler ( | ||||
| 	output [2:0] debug_state, | ||||
| 	output reg [7:0] opcode, | ||||
| 	output  reg       req, | ||||
| 	input   clock, | ||||
| 	input   index, | ||||
| 	input   rdata, | ||||
| 	input   reset, | ||||
| 	input   sampleclock | ||||
| ); | ||||
|  | ||||
| //`#start body` -- edit after this line, do not edit this line | ||||
|  | ||||
| // NOTE: Reset pulse is used in both clock domains, and must be long enough | ||||
| // to be detected in both. | ||||
|  | ||||
| reg [5:0] counter; | ||||
|  | ||||
| reg index_edge; | ||||
| reg rdata_edge; | ||||
|  | ||||
| reg req_toggle; | ||||
|  | ||||
| reg rdata_toggle; | ||||
| reg old_rdata_toggle; | ||||
|  | ||||
| reg index_toggle; | ||||
| reg old_index_toggle; | ||||
|  | ||||
| always @(posedge rdata) | ||||
| begin | ||||
|     rdata_toggle <= ~rdata_toggle; | ||||
| end | ||||
|  | ||||
| always @(posedge index) | ||||
| begin | ||||
|     index_toggle <= ~index_toggle; | ||||
| end | ||||
|  | ||||
| always @(posedge sampleclock) | ||||
| begin | ||||
|     if (reset) | ||||
|     begin | ||||
|         old_rdata_toggle <= 0; | ||||
|         old_index_toggle <= 0; | ||||
|          | ||||
|         index_edge <= 0; | ||||
|         rdata_edge <= 0; | ||||
|         counter <= 0; | ||||
|         req_toggle <= 0; | ||||
|     end | ||||
|     else | ||||
|     begin | ||||
|         /* If data_toggle or index_toggle have changed state, this means that they've | ||||
|          * gone high since the last sampleclock. */ | ||||
|           | ||||
|         index_edge <= index_toggle != old_index_toggle; | ||||
|         old_index_toggle <= index_toggle; | ||||
|          | ||||
|         rdata_edge <= rdata_toggle != old_rdata_toggle; | ||||
|         old_rdata_toggle <= rdata_toggle; | ||||
|          | ||||
|         if (rdata_edge || index_edge || (counter == 6'h3f)) begin | ||||
|             opcode <= { rdata_edge, index_edge, counter }; | ||||
|             req_toggle <= ~req_toggle; | ||||
|             counter <= 1;   /* remember to count this tick */ | ||||
|         end else begin | ||||
|             counter <= counter + 1; | ||||
|         end | ||||
|     end | ||||
| end | ||||
|  | ||||
| reg req_toggle_q; | ||||
|  | ||||
| always @(posedge clock) | ||||
| begin | ||||
|     if (reset) begin | ||||
|         req_toggle_q <= 0; | ||||
|         req <= 0; | ||||
|     end else begin | ||||
|         req_toggle_q <= req_toggle; | ||||
|         req <= (req_toggle != req_toggle_q); | ||||
|     end | ||||
| end | ||||
|  | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
| endmodule | ||||
| //`#start footer` -- edit after this line, do not edit this line | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										95
									
								
								FluxEngine.cydsn/Sequencer/Sequencer.v
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										95
									
								
								FluxEngine.cydsn/Sequencer/Sequencer.v
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,95 @@ | ||||
|  | ||||
| //`#start header` -- edit after this line, do not edit this line | ||||
| `include "cypress.v" | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
| // Generated on 11/24/2019 at 17:25 | ||||
| // Component: Sequencer | ||||
| module Sequencer ( | ||||
| 	output req, /* request new data on leading edge */ | ||||
| 	output wdata, | ||||
|     output [2:0] debug_state, | ||||
| 	input clock, | ||||
| 	input dataclock, /* incoming data on leading edge */ | ||||
| 	input [7:0] opcode, | ||||
|     input index, | ||||
|     input sampleclock, | ||||
|     input reset | ||||
| ); | ||||
|  | ||||
| //`#start body` -- edit after this line, do not edit this line | ||||
|  | ||||
| localparam STATE_LOAD = 0; | ||||
| localparam STATE_WRITING = 1; | ||||
|  | ||||
| reg state; | ||||
| reg [5:0] countdown; | ||||
| reg pulsepending; | ||||
|  | ||||
| assign req = (!reset && (state == STATE_LOAD)); | ||||
| assign wdata = (!reset && (state == STATE_WRITING) && (countdown == 0) && pulsepending); | ||||
| assign debug_state = 0; | ||||
|  | ||||
| reg olddataclock; | ||||
| wire dataclocked; | ||||
| always @(posedge clock) olddataclock <= dataclock; | ||||
| assign dataclocked = !olddataclock && dataclock; | ||||
|  | ||||
| reg oldsampleclock; | ||||
| reg sampleclocked; | ||||
|  | ||||
| reg oldindex; | ||||
| wire indexed; | ||||
| always @(posedge clock) oldindex <= index; | ||||
| assign indexed = !oldindex && index; | ||||
|  | ||||
| always @(posedge clock) | ||||
| begin | ||||
|     if (reset) | ||||
|     begin | ||||
|         state <= STATE_LOAD; | ||||
|         countdown <= 0; | ||||
|         pulsepending <= 0; | ||||
|         oldsampleclock <= 0; | ||||
|     end | ||||
|     else | ||||
|     begin | ||||
|         if (!oldsampleclock && sampleclock) | ||||
|             sampleclocked <= 1; | ||||
|         oldsampleclock <= sampleclock; | ||||
|              | ||||
|         case (state) | ||||
|             STATE_LOAD: | ||||
|             begin | ||||
|                 /* A posedge on dataclocked indicates that another opcode has | ||||
|                  * arrived. */ | ||||
|                 if (dataclocked) | ||||
|                 begin | ||||
|                     pulsepending <= opcode[7]; | ||||
|                     if (opcode[5:0] == 0) | ||||
|                         countdown <= 0; | ||||
|                     else | ||||
|                         countdown <= opcode[5:0] - 1; /* compensate for extra tick in state machine */ | ||||
|                      | ||||
|                     state <= STATE_WRITING; | ||||
|                 end | ||||
|             end | ||||
|              | ||||
|             STATE_WRITING: | ||||
|             begin | ||||
|                 if (sampleclocked) | ||||
|                 begin | ||||
|                     if (countdown == 0) | ||||
|                         state <= STATE_LOAD; | ||||
|                     else | ||||
|                         countdown <= countdown - 1; | ||||
|                     sampleclocked <= 0; | ||||
|                 end | ||||
|             end | ||||
|         endcase | ||||
|     end | ||||
| end | ||||
|  | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
| endmodule | ||||
| //`#start footer` -- edit after this line, do not edit this line | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
							
								
								
									
										
											BIN
										
									
								
								FluxEngine.cydsn/SuperCounter/SuperCounter.cysym
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								FluxEngine.cydsn/SuperCounter/SuperCounter.cysym
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										156
									
								
								FluxEngine.cydsn/SuperCounter/SuperCounter.v
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										156
									
								
								FluxEngine.cydsn/SuperCounter/SuperCounter.v
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,156 @@ | ||||
|  | ||||
| //`#start header` -- edit after this line, do not edit this line | ||||
| `include "cypress.v" | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
| // Generated on 11/16/2017 at 15:44 | ||||
| // Component: FIFOout | ||||
| module SuperCounter ( | ||||
| 	input clk, | ||||
|     input reset, | ||||
|     input count, | ||||
|     output [7:0] d, | ||||
| 	output drq, | ||||
| 	output empty, | ||||
|     output ack | ||||
| ); | ||||
|  | ||||
| //`#start body` -- edit after this line, do not edit this line | ||||
|  | ||||
| parameter ResetValue = 0; | ||||
| parameter Delta = 1; | ||||
|      | ||||
| wire [7:0] po; | ||||
| assign d = po; | ||||
|  | ||||
| localparam STATE_RESET = 0; | ||||
| localparam STATE_WAIT = 1; | ||||
| localparam STATE_ADD = 2; | ||||
|  | ||||
| reg oldcount; | ||||
| wire counted; | ||||
| assign counted = count && !oldcount; | ||||
|  | ||||
| always @(posedge clk) oldcount <= count; | ||||
|  | ||||
| wire [2:0] cs; | ||||
| assign cs = reset ? STATE_RESET : (counted ? STATE_ADD : STATE_WAIT); | ||||
|              | ||||
| cy_psoc3_dp #(.d0_init(ResetValue), .d1_init(Delta),  | ||||
| .cy_dpconfig( | ||||
| { | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC___D0, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM0:  STATE_RESET*/ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC__ALU, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM1:  STATE_WAIT*/ | ||||
|     `CS_ALU_OP__ADD, `CS_SRCA_A0, `CS_SRCB_D1, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC__ALU, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM2:  STATE_ADD*/ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM3:             */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM4:             */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM5:             */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM6:             */ | ||||
|     `CS_ALU_OP_PASS, `CS_SRCA_A0, `CS_SRCB_D0, | ||||
|     `CS_SHFT_OP_PASS, `CS_A0_SRC_NONE, `CS_A1_SRC_NONE, | ||||
|     `CS_FEEDBACK_DSBL, `CS_CI_SEL_CFGA, `CS_SI_SEL_CFGA, | ||||
|     `CS_CMP_SEL_CFGA, /*CFGRAM7:             */ | ||||
|     8'hFF, 8'h00,  /*CFG9:             */ | ||||
|     8'hFF, 8'hFF,  /*CFG11-10:             */ | ||||
|     `SC_CMPB_A1_D1, `SC_CMPA_A1_D1, `SC_CI_B_ARITH, | ||||
|     `SC_CI_A_ARITH, `SC_C1_MASK_DSBL, `SC_C0_MASK_DSBL, | ||||
|     `SC_A_MASK_DSBL, `SC_DEF_SI_0, `SC_SI_B_DEFSI, | ||||
|     `SC_SI_A_DEFSI, /*CFG13-12:             */ | ||||
|     `SC_A0_SRC_ACC, `SC_SHIFT_SL, 1'h0, | ||||
|     1'h0, `SC_FIFO1_BUS, `SC_FIFO0_BUS, | ||||
|     `SC_MSB_DSBL, `SC_MSB_BIT0, `SC_MSB_NOCHN, | ||||
|     `SC_FB_NOCHN, `SC_CMP1_NOCHN, | ||||
|     `SC_CMP0_NOCHN, /*CFG15-14:             */ | ||||
|     10'h00, `SC_FIFO_CLK__DP,`SC_FIFO_CAP_AX, | ||||
|     `SC_FIFO_LEVEL,`SC_FIFO_ASYNC,`SC_EXTCRC_DSBL, | ||||
|     `SC_WRK16CAT_DSBL /*CFG17-16:             */ | ||||
| } | ||||
| )) dp( | ||||
|         /*  input                   */  .reset(1'b0), | ||||
|         /*  input                   */  .clk(clk), | ||||
|         /*  input   [02:00]         */  .cs_addr(cs), | ||||
|         /*  input                   */  .route_si(1'b0), | ||||
|         /*  input                   */  .route_ci(1'b0), | ||||
|         /*  input                   */  .f0_load(1'b0), | ||||
|         /*  input                   */  .f1_load(1'b0), | ||||
|         /*  input                   */  .d0_load(1'b0), | ||||
|         /*  input                   */  .d1_load(1'b0), | ||||
|         /*  output                  */  .ce0(), | ||||
|         /*  output                  */  .cl0(), | ||||
|         /*  output                  */  .z0(), | ||||
|         /*  output                  */  .ff0(), | ||||
|         /*  output                  */  .ce1(), | ||||
|         /*  output                  */  .cl1(), | ||||
|         /*  output                  */  .z1(), | ||||
|         /*  output                  */  .ff1(), | ||||
|         /*  output                  */  .ov_msb(), | ||||
|         /*  output                  */  .co_msb(), | ||||
|         /*  output                  */  .cmsb(), | ||||
|         /*  output                  */  .so(), | ||||
|         /*  output                  */  .f0_bus_stat(), | ||||
|         /*  output                  */  .f0_blk_stat(), | ||||
|         /*  output                  */  .f1_bus_stat(), | ||||
|         /*  output                  */  .f1_blk_stat(), | ||||
|          | ||||
|         /* input                    */  .ci(1'b0),     // Carry in from previous stage | ||||
|         /* output                   */  .co(),// Carry out to next stage | ||||
|         /* input                    */  .sir(1'b0),    // Shift in from right side | ||||
|         /* output                   */  .sor(),        // Shift out to right side | ||||
|         /* input                    */  .sil(1'b0),    // Shift in from left side | ||||
|         /* output                   */  .sol(),        // Shift out to left side | ||||
|         /* input                    */  .msbi(1'b0),   // MSB chain in | ||||
|         /* output                   */  .msbo(),       // MSB chain out | ||||
|         /* input [01:00]            */  .cei(2'b0),    // Compare equal in from prev stage | ||||
|         /* output [01:00]           */  .ceo(),        // Compare equal out to next stage | ||||
|         /* input [01:00]            */  .cli(2'b0),    // Compare less than in from prv stage | ||||
|         /* output [01:00]           */  .clo(),        // Compare less than out to next stage | ||||
|         /* input [01:00]            */  .zi(2'b0),     // Zero detect in from previous stage | ||||
|         /* output [01:00]           */  .zo(),         // Zero detect out to next stage | ||||
|         /* input [01:00]            */  .fi(2'b0),     // 0xFF detect in from previous stage | ||||
|         /* output [01:00]           */  .fo(),         // 0xFF detect out to next stage | ||||
|         /* input [01:00]            */  .capi(2'b0),   // Software capture from previous stage | ||||
|         /* output [01:00]           */  .capo(),       // Software capture to next stage | ||||
|         /* input                    */  .cfbi(1'b0),   // CRC Feedback in from previous stage | ||||
|         /* output                   */  .cfbo(),       // CRC Feedback out to next stage | ||||
|         /* input [07:00]            */  .pi(8'b0),     // Parallel data port | ||||
|         /* output [07:00]           */  .po(po)       // Parallel data port | ||||
| ); | ||||
|  | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
| endmodule | ||||
| //`#start footer` -- edit after this line, do not edit this line | ||||
| //`#end` -- edit above this line, do not edit this line | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,18 +0,0 @@ | ||||
| const READ = 1 | ||||
| const WRITE = 2 | ||||
|  | ||||
| filename = "Generated_Source\PSoC5\USBFS_descr.c" | ||||
| set fso = CreateObject("Scripting.FileSystemObject") | ||||
|  | ||||
| set file = fso.OpenTextFile(filename, READ) | ||||
| text = file.ReadAll | ||||
| file.Close | ||||
|  | ||||
| set r = New RegExp | ||||
| r.MultiLine = True | ||||
| r.Pattern = "/\* +compatibleID.*\n.*" | ||||
| text = r.replace(text, "'W', 'I', 'N', 'U', 'S', 'B', 0, 0,") | ||||
|  | ||||
| set file = fso.CreateTextFile(filename, True) | ||||
| file.Write text | ||||
| file.Close | ||||
							
								
								
									
										76
									
								
								Makefile
									
									
									
									
									
								
							
							
						
						
									
										76
									
								
								Makefile
									
									
									
									
									
								
							| @@ -1,33 +1,89 @@ | ||||
| PACKAGES = zlib sqlite3 libusb-1.0 | ||||
| PACKAGES = zlib sqlite3 protobuf | ||||
| ifeq ($(shell uname),Linux) | ||||
| PACKAGES += gtk+-3.0 | ||||
| endif | ||||
|  | ||||
| export CFLAGS = -O3 -g --std=c++14 \ | ||||
| export CFLAGS = \ | ||||
| 	-ffunction-sections -fdata-sections | ||||
| export LDFLAGS = -O3 | ||||
| export CXXFLAGS = $(CFLAGS) \ | ||||
| 	--std=gnu++2a \ | ||||
| 	-Wno-deprecated-enum-enum-conversion \ | ||||
| 	-Wno-deprecated-enum-float-conversion | ||||
| export LDFLAGS = -pthread | ||||
|  | ||||
| export COPTFLAGS = -Os | ||||
| export LDOPTFLAGS = -Os | ||||
|  | ||||
| export CDBGFLAGS = -O0 -g | ||||
| export LDDBGFLAGS = -O0 -g | ||||
|  | ||||
| ifeq ($(OS), Windows_NT) | ||||
| else | ||||
| ifeq ($(shell uname),Darwin) | ||||
| else | ||||
| 	PACKAGES += libudev | ||||
| endif | ||||
| endif | ||||
|  | ||||
| ifeq ($(OS), Windows_NT) | ||||
| export PROTOC = /mingw32/bin/protoc | ||||
| export CC = /mingw32/bin/gcc | ||||
| export CXX = /mingw32/bin/g++ | ||||
| export AR = /mingw32/bin/ar rcs | ||||
| export AR = /mingw32/bin/ar rc | ||||
| export RANLIB = /mingw32/bin/ranlib | ||||
| export STRIP = /mingw32/bin/strip | ||||
| export CFLAGS += -I/mingw32/include/libusb-1.0 | ||||
| export WINDRES = /mingw32/bin/windres | ||||
| export CFLAGS += -I/mingw32/include/libusb-1.0 -I/mingw32/include | ||||
| export LDFLAGS += | ||||
| export LIBS = -static -lz -lsqlite3 -lusb-1.0 | ||||
| export LIBS += -L/mingw32/lib -static -lz -lsqlite3 \ | ||||
| 	-lsetupapi -lwinusb -lole32 -lprotobuf -luuid | ||||
| export GUILIBS += -luser32 -lkernel32 -lgdi32 -lcomctl32 -luxtheme -lmsimg32 \ | ||||
| 	-lcomdlg32 -ld2d1 -ldwrite -lole32 -loleaut32 -loleacc -luuid \ | ||||
| 	-lwindowscodecs | ||||
| export EXTENSION = .exe | ||||
| else | ||||
|  | ||||
| packages-exist = $(shell pkg-config --exists $(PACKAGES) && echo yes) | ||||
| ifneq ($(packages-exist),yes) | ||||
| $(warning These pkg-config packages are installed: $(shell pkg-config --list-all | sort | awk '{print $$1}')) | ||||
| $(error You must have these pkg-config packages installed: $(PACKAGES)) | ||||
| endif | ||||
|  | ||||
| export PROTOC = protoc | ||||
| export CC = gcc | ||||
| export CXX = g++ | ||||
| export AR = ar rcs | ||||
| export COBJC = clang | ||||
| export AR = ar rc | ||||
| export RANLIB = ranlib | ||||
| export STRIP = strip | ||||
| export CFLAGS += $(shell pkg-config --cflags $(PACKAGES)) | ||||
| export LDFLAGS += | ||||
| export LIBS = $(shell pkg-config --libs $(PACKAGES)) | ||||
| export LIBS += $(shell pkg-config --libs $(PACKAGES)) -ldl | ||||
| export EXTENSION = | ||||
|  | ||||
| ifeq ($(shell uname),Darwin) | ||||
| AR = ar rcS | ||||
| RANLIB += -c -no_warning_for_no_symbols | ||||
| export CC = clang | ||||
| export CXX = clang++ | ||||
| export COBJC = clang | ||||
| export LDFLAGS += \ | ||||
| 	-framework Foundation \ | ||||
| 	-framework AppKit \ | ||||
| 	-framework IOKit \ | ||||
| 	-framework CoreFoundation | ||||
| endif | ||||
|  | ||||
| CFLAGS += -Ilib -Idep/fmt | ||||
| endif | ||||
| export XXD = xxd | ||||
|  | ||||
| CFLAGS += -Ilib -Idep/fmt -Iarch | ||||
|  | ||||
| export OBJDIR = .obj | ||||
|  | ||||
| all: .obj/build.ninja | ||||
| 	@ninja -f .obj/build.ninja -v | ||||
| 	@ninja -f .obj/build.ninja -k 0 | ||||
| 	@if command -v cscope > /dev/null; then cscope -bRq; fi | ||||
|  | ||||
| clean: | ||||
| 	@echo CLEAN | ||||
|   | ||||
							
								
								
									
										105
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										105
									
								
								README.md
									
									
									
									
									
								
							| @@ -4,6 +4,11 @@ FluxEngine | ||||
| (If you're reading this on GitHub, the formatting's a bit messed up. [Try the | ||||
| version on cowlark.com instead.](http://cowlark.com/fluxengine/) | ||||
|  | ||||
| **Breaking news!** As of 2021-05-21, the command line environment has changed | ||||
| _substantially_ (to make it more consistent and flexible, and allow some new | ||||
| backend features like multi-format IBM scheme disks, which are popular with | ||||
| CP/M). If things don't work the way you expect, please check the documentation. | ||||
|  | ||||
| What? | ||||
| ----- | ||||
|  | ||||
| @@ -24,17 +29,18 @@ Don't believe me? Watch the demo reel! | ||||
| <iframe width="373" height="210" src="https://www.youtube.com/embed/m_s1iw8eW7o" frameborder="0" allow="accelerometer; autoplay; encrypted-media; gyroscope; picture-in-picture" allowfullscreen></iframe> | ||||
| </div> | ||||
|  | ||||
| **Important note.** On 2019-02-09 I did a hardware redesign and moved the pins on | ||||
| the board. Sorry for the inconvenience, but it means you don't have to modify | ||||
| the board any more to make it work. If you built the hardware prior to then, | ||||
| you'll need to adjust it. | ||||
| **New!** The FluxEngine client software now works with | ||||
| [GreaseWeazle](https://github.com/keirf/Greaseweazle/wiki) hardware. So, if you | ||||
| can't find a PSoC5 development kit, or don't want to use the Cypress Windows | ||||
| tools for programming it, you can use one of these instead. Very nearly all | ||||
| FluxEngine features are available with the GreaseWeazle and it works out-of-the | ||||
| box. See the [dedicated GreaseWeazle documentation page](doc/greaseweazle.md) | ||||
| for more information. | ||||
|  | ||||
| **Another important note.** On 2019-07-03 I've revamped the build process and | ||||
| the (command line) user interface. It should be much nicer now, not least in | ||||
| that there's a single client binary with all the functionality in it. The | ||||
| interface is a little different, but not much. The build process is now | ||||
| better (simpler). See [the building](doc/building.md) and | ||||
| [using](doc/using.md) pages for more details. | ||||
| **Important note.** On 2020-04-02 I changed the bytecode format (and firmware). | ||||
| Flux files will need to be upgraded with `fluxengine upgradefluxfile`. The new | ||||
| format should be more reliable and use way, way less bandwidth. Sorry for the | ||||
| inconvenience. | ||||
|  | ||||
| Where? | ||||
| ------ | ||||
| @@ -59,8 +65,15 @@ following friendly articles: | ||||
|   - [Using a FluxEngine](doc/using.md) ∾ what to do with your new hardware ∾ | ||||
|     flux files and image files ∾ knowing what you're doing | ||||
|  | ||||
|   - [Troubleshooting dubious disks](doc/problems.md) ∾ it's not an exact science ∾ | ||||
|     the sector map ∾ clock detection and the histogram | ||||
|   - [Using GreaseWeazle hardware with the FluxEngine client | ||||
| 	software](doc/greaseweazle.md) ∾ what works ∾ what doesn't work ∾ where to | ||||
| 	go for help | ||||
|  | ||||
|   - [Troubleshooting dubious disks](doc/problems.md) ∾ it's not an exact | ||||
| 	science ∾ the sector map ∾ clock detection and the histogram | ||||
|  | ||||
|   - [Checking your drive](doc/driveresponse.md) ∾ you can't do that with that ∾ | ||||
| 	measuring your drive's ability to work with exotic formats | ||||
|  | ||||
| Which? | ||||
| ------ | ||||
| @@ -77,22 +90,26 @@ people who've had it work). | ||||
|  | ||||
| ### Old disk formats | ||||
|  | ||||
| | Format                                   | Read? | Write? | Notes | | ||||
| |:-----------------------------------------|:-----:|:------:|-------| | ||||
| | IBM PC compatible                        |  🦄   |        | and compatibles (like the Atari ST) | | ||||
| | [Acorn ADFS](doc/disk-acornadfs.md)      |  🦄   |        | single- and double- sided           | | ||||
| | [Acorn DFS](doc/disk-acorndfs.md)        |  🦄   |        |                                     | | ||||
| | [Ampro Little Board](doc/disk-ampro.md)  |  🦖   |        |                                     | | ||||
| | [Apple II DOS 3.3](doc/disk-apple2.md)   |  🦄   |        | doesn't do logical sector remapping | | ||||
| | [Amiga](doc/disk-amiga.md)               |  🦄   |        |                                     | | ||||
| | [Commodore 64 1541](doc/disk-c64.md)     |  🦖   |        | and probably the other GCR formats  | | ||||
| | [Brother 120kB](doc/disk-brother.md)     |  🦄   |        |                                     | | ||||
| | [Brother 240kB](doc/disk-brother.md)     |  🦄   |   🦄   |                                     | | ||||
| | [Brother FB-100](doc/disk-fb100.md)      |  🦖   |        | Tandy Model 100, Husky Hunter, knitting machines | | ||||
| | [Macintosh 800kB](doc/disk-macintosh.md) |  🦖   |        | and probably the 400kB too          | | ||||
| | [TRS-80](doc/disk-trs80.md)              |  🦖   |        | a minor variation of the IBM scheme | | ||||
| | Format                                    | Read? | Write? | Notes | | ||||
| |:------------------------------------------|:-----:|:------:|-------| | ||||
| | [IBM PC compatible](doc/disk-ibm.md)      |  🦄   |   🦄   | and compatibles (like the Atari ST) | | ||||
| | [Acorn ADFS](doc/disk-acornadfs.md)       |  🦄   |   🦖*  | single- and double- sided           | | ||||
| | [Acorn DFS](doc/disk-acorndfs.md)         |  🦄   |   🦖*  |                                     | | ||||
| | [Ampro Little Board](doc/disk-ampro.md)   |  🦖   |   🦖*  |                                     | | ||||
| | [Apple II DOS 3.3](doc/disk-apple2.md)    |  🦄   |        | doesn't do logical sector remapping | | ||||
| | [Amiga](doc/disk-amiga.md)                |  🦄   |   🦄   |                                     | | ||||
| | [Commodore 64 1541/1581](doc/disk-c64.md) |  🦄   |   🦄   | and probably the other formats      | | ||||
| | [Brother 120kB](doc/disk-brother.md)      |  🦄   |   🦖   |                                     | | ||||
| | [Brother 240kB](doc/disk-brother.md)      |  🦄   |   🦄   |                                     | | ||||
| | [Brother FB-100](doc/disk-fb100.md)       |  🦖   |        | Tandy Model 100, Husky Hunter, knitting machines | | ||||
| | [Macintosh 800kB](doc/disk-macintosh.md)  |  🦄   |   🦄   | and probably the 400kB too          | | ||||
| | [TRS-80](doc/disk-trs80.md)               |  🦖   |   🦖*  | a minor variation of the IBM scheme | | ||||
| {: .datatable } | ||||
|  | ||||
| `*`: these formats are variations of the generic IBM format, and since the | ||||
| IBM writer is completely generic, it should be configurable for these | ||||
| formats... theoretically. I don't have the hardware to try it. | ||||
|  | ||||
| ### Even older disk formats | ||||
|  | ||||
| These formats are for particularly old, weird architectures, even by the | ||||
| @@ -105,8 +122,13 @@ at least, check the CRC so what data's there is probably good. | ||||
| |:-----------------------------------------|:-----:|:------:|-------| | ||||
| | [AES Superplus / No Problem](doc/disk-aeslanier.md) |  🦖   | | hard sectors! | | ||||
| | [Durango F85](doc/disk-durangof85.md)    |  🦖   |        | 5.25" | | ||||
| | [Victor 9000](doc/disk-victor9k.md)      |  🦖   |        | 8-inch        | | ||||
| | [Zilog MCZ](doc/disk-zilogmcz.md)        |  🦖   |        | 8-inch _and_ hard sectors | | ||||
| | [DVK MX](doc/disk-mx.md)                 |  🦖   |        | Soviet PDP-11 clone | | ||||
| | [VDS Eco1](doc/disk-eco1.md)             |  🦖   |        | 8" mixed format | | ||||
| | [Micropolis](doc/disk-micropolis.md)     |  🦄   |        | Micropolis 100tpi drives | | ||||
| | [Northstar(doc/disk-northstar.md)        |  🦖   |   🦖   | 5.25" hard sectors | | ||||
| | [TI DS990 FD1000](doc/disk-tids990.md)   |  🦄   |  🦄    | 8" | | ||||
| | [Victor 9000](doc/disk-victor9k.md)      |  🦖   |        | 8" | | ||||
| | [Zilog MCZ](doc/disk-zilogmcz.md)        |  🦖   |        | 8" _and_ hard sectors | | ||||
| {: .datatable } | ||||
|  | ||||
| ### Notes | ||||
| @@ -189,7 +211,30 @@ Jonathan Müller (`foonathan <https://github.com/foonathan>`) with | ||||
| contributions from many other people. It is licensed under the terms of the | ||||
| BSD license. Please see the contents of the directory for the full text. | ||||
|  | ||||
| As an exception, `dep/fnmatchemu` contains parts of the OpenBSD C library | ||||
| code, Todd Miller and William A. Rowe (and probably others). It is licensed | ||||
| As an exception, `dep/emu` contains parts of the OpenBSD C library | ||||
| code, maintained by Todd Miller and William A. Rowe (and probably others). It is licensed | ||||
| under the terms of the 3-clause BSD license. Please see the contents of the | ||||
| directory for the full text. It's been lightly modified by me. | ||||
|  | ||||
| As an exception, `dep/agg` contains parts of the Anti-Grain Antialiasing | ||||
| library, written by Maxim Semanarev (and others). It is licensed under the | ||||
| terms of the 3-clause BSD license. Please see the contents of the directory for | ||||
| the full text. It's been lightly modified by me. | ||||
|  | ||||
| As an exception, `dep/stb` contains parts of the libstb utility library, | ||||
| written by Sean T Barett (and others). It is public domain/Unlicense/MIT | ||||
| licensed, at your choice. Please see the contents of the directory for the full | ||||
| text. | ||||
|  | ||||
| As an exception, `dep/snowhouse` contains the snowhouse assertion library, | ||||
| taken from https://github.com/banditcpp/snowhouse. It is Boost Standard License | ||||
| 1.0 licensed. Please see the contents of the directory for the full text. | ||||
|  | ||||
| As an exception, `dep/libusbp` contains the libusbp library, taken from | ||||
| https://github.com/pololu/libusbp. It is MIT licensed. Please see the contents | ||||
| of the directory for the full text. | ||||
|  | ||||
| As an exception, `dep/libui` contains the libui GUI library, taken from | ||||
| https://github.com/andlabs/libui. It is MIT licensed. Please see the contents | ||||
| of the directory for the full text. | ||||
|  | ||||
|   | ||||
| @@ -5,16 +5,6 @@ | ||||
| #define AESLANIER_SECTOR_LENGTH    256 | ||||
| #define AESLANIER_RECORD_SIZE      (AESLANIER_SECTOR_LENGTH + 5) | ||||
| 
 | ||||
| class Sector; | ||||
| class Fluxmap; | ||||
| 
 | ||||
| class AesLanierDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
|     virtual ~AesLanierDecoder() {} | ||||
| 
 | ||||
|     RecordType advanceToNextRecord(); | ||||
|     void decodeSectorRecord(); | ||||
| }; | ||||
| extern std::unique_ptr<AbstractDecoder> createAesLanierDecoder(const DecoderProto& config); | ||||
| 
 | ||||
| #endif | ||||
							
								
								
									
										4
									
								
								arch/aeslanier/aeslanier.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								arch/aeslanier/aeslanier.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| message AesLanierDecoderProto {} | ||||
|  | ||||
							
								
								
									
										79
									
								
								arch/aeslanier/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								arch/aeslanier/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,79 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "aeslanier.h" | ||||
| #include "crc.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "sector.h" | ||||
| #include "bytes.h" | ||||
| #include "fmt/format.h" | ||||
| #include <string.h> | ||||
|  | ||||
| static const FluxPattern SECTOR_PATTERN(32, AESLANIER_RECORD_SEPARATOR); | ||||
|  | ||||
| /* This is actually M2FM, rather than MFM, but it our MFM/FM decoder copes fine with it. */ | ||||
|  | ||||
| static Bytes reverse_bits(const Bytes& input) | ||||
| { | ||||
|     Bytes output; | ||||
|     ByteWriter bw(output); | ||||
|  | ||||
|     for (uint8_t b : input) | ||||
|         bw.write_8(reverse_bits(b)); | ||||
|     return output; | ||||
| } | ||||
|  | ||||
| class AesLanierDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	AesLanierDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
|  | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		_sector->clock = _fmr->seekToPattern(SECTOR_PATTERN); | ||||
| 		if (_fmr->eof() || !_sector->clock) | ||||
| 			return UNKNOWN_RECORD; | ||||
| 		return SECTOR_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		/* Skip ID mark. */ | ||||
|  | ||||
| 		readRawBits(16); | ||||
|  | ||||
| 		const auto& rawbits = readRawBits(AESLANIER_RECORD_SIZE*16); | ||||
| 		const auto& bytes = decodeFmMfm(rawbits).slice(0, AESLANIER_RECORD_SIZE); | ||||
| 		const auto& reversed = reverse_bits(bytes); | ||||
|  | ||||
| 		_sector->logicalTrack = reversed[1]; | ||||
| 		_sector->logicalSide = 0; | ||||
| 		_sector->logicalSector = reversed[2]; | ||||
|  | ||||
| 		/* Check header 'checksum' (which seems far too simple to mean much). */ | ||||
|  | ||||
| 		{ | ||||
| 			uint8_t wanted = reversed[3]; | ||||
| 			uint8_t got = reversed[1] + reversed[2]; | ||||
| 			if (wanted != got) | ||||
| 				return; | ||||
| 		} | ||||
|  | ||||
| 		/* Check data checksum, which also includes the header and is | ||||
| 			* significantly better. */ | ||||
|  | ||||
| 		_sector->data = reversed.slice(1, AESLANIER_SECTOR_LENGTH); | ||||
| 		uint16_t wanted = reversed.reader().seek(0x101).read_le16(); | ||||
| 		uint16_t got = crc16ref(MODBUS_POLY_REF, _sector->data); | ||||
| 		_sector->status = (wanted == got) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createAesLanierDecoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new AesLanierDecoder(config)); | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										100
									
								
								arch/amiga/amiga.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										100
									
								
								arch/amiga/amiga.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,100 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "amiga.h" | ||||
| #include "bytes.h" | ||||
| #include "fmt/format.h" | ||||
|  | ||||
| uint32_t amigaChecksum(const Bytes& bytes) | ||||
| { | ||||
|     ByteReader br(bytes); | ||||
|     uint32_t checksum = 0; | ||||
|  | ||||
|     assert((bytes.size() & 3) == 0); | ||||
|     while (!br.eof()) | ||||
|         checksum ^= br.read_be32(); | ||||
|  | ||||
|     return checksum & 0x55555555; | ||||
| } | ||||
|  | ||||
| static uint8_t everyother(uint16_t x) | ||||
| { | ||||
| 	                  /* aabb ccdd eeff gghh */ | ||||
| 	x &= 0x6666;      /* 0ab0 0cd0 0ef0 0gh0 */ | ||||
| 	x >>= 1;          /* 00ab 00cd 00ef 00gh */ | ||||
| 	x |= x << 2;      /* abab cdcd efef ghgh */ | ||||
| 	x &= 0x3c3c;      /* 00ab cd00 00ef gh00 */ | ||||
| 	x >>= 2;          /* 0000 abcd 0000 efgh */ | ||||
| 	x |= x >> 4;      /* 0000 abcd abcd efgh */ | ||||
| 	return x; | ||||
| } | ||||
|  | ||||
| Bytes amigaInterleave(const Bytes& input) | ||||
| { | ||||
| 	Bytes output; | ||||
| 	ByteWriter bw(output); | ||||
|  | ||||
| 	/* Write all odd bits. (Numbering starts at 0...) */ | ||||
|  | ||||
| 	{ | ||||
| 		ByteReader br(input); | ||||
| 		while (!br.eof()) | ||||
| 		{ | ||||
| 			uint16_t x = br.read_be16(); | ||||
| 			x &= 0xaaaa;       /* a0b0 c0d0 e0f0 g0h0 */ | ||||
| 			x |= x >> 1;       /* aabb ccdd eeff gghh */ | ||||
| 			x = everyother(x); /* 0000 0000 abcd efgh */ | ||||
| 			bw.write_8(x); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	/* Write all even bits. */ | ||||
|  | ||||
| 	{ | ||||
| 		ByteReader br(input); | ||||
| 		while (!br.eof()) | ||||
| 		{ | ||||
| 			uint16_t x = br.read_be16(); | ||||
| 			x &= 0x5555;       /* 0a0b 0c0d 0e0f 0g0h */ | ||||
| 			x |= x << 1;       /* aabb ccdd eeff gghh */ | ||||
| 			x = everyother(x); /* 0000 0000 abcd efgh */ | ||||
| 			bw.write_8(x); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	return output; | ||||
| } | ||||
|  | ||||
| Bytes amigaDeinterleave(const uint8_t*& input, size_t len) | ||||
| { | ||||
|     assert(!(len & 1)); | ||||
|     const uint8_t* odds = &input[0]; | ||||
|     const uint8_t* evens = &input[len/2]; | ||||
|     Bytes output; | ||||
|     ByteWriter bw(output); | ||||
|  | ||||
|     for (size_t i=0; i<len/2; i++) | ||||
|     { | ||||
|         uint8_t o = *odds++; | ||||
|         uint8_t e = *evens++; | ||||
|  | ||||
|         /* This is the 'Interleave bits with 64-bit multiply' technique from | ||||
|          * http://graphics.stanford.edu/~seander/bithacks.html#InterleaveBMN | ||||
|          */ | ||||
|         uint16_t result = | ||||
|             (((e * 0x0101010101010101ULL & 0x8040201008040201ULL) | ||||
|                 * 0x0102040810204081ULL >> 49) & 0x5555) | | ||||
|             (((o * 0x0101010101010101ULL & 0x8040201008040201ULL) | ||||
|                 * 0x0102040810204081ULL >> 48) & 0xAAAA); | ||||
|          | ||||
|         bw.write_be16(result); | ||||
|     } | ||||
|  | ||||
|     input += len; | ||||
|     return output; | ||||
| } | ||||
|  | ||||
| Bytes amigaDeinterleave(const Bytes& input) | ||||
| { | ||||
| 	const uint8_t* ptr = input.cbegin(); | ||||
| 	return amigaDeinterleave(ptr, input.size()); | ||||
| } | ||||
							
								
								
									
										20
									
								
								arch/amiga/amiga.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								arch/amiga/amiga.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| #ifndef AMIGA_H | ||||
| #define AMIGA_H | ||||
|  | ||||
| #include "encoders/encoders.h" | ||||
|  | ||||
| #define AMIGA_SECTOR_RECORD 0xaaaa44894489LL | ||||
|  | ||||
| #define AMIGA_TRACKS_PER_DISK 80 | ||||
| #define AMIGA_SECTORS_PER_TRACK 11 | ||||
| #define AMIGA_RECORD_SIZE 0x21f | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createAmigaDecoder(const DecoderProto& config); | ||||
| extern std::unique_ptr<AbstractEncoder> createAmigaEncoder(const EncoderProto& config); | ||||
|  | ||||
| extern uint32_t amigaChecksum(const Bytes& bytes); | ||||
| extern Bytes amigaInterleave(const Bytes& input); | ||||
| extern Bytes amigaDeinterleave(const uint8_t*& input, size_t len); | ||||
| extern Bytes amigaDeinterleave(const Bytes& input); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										13
									
								
								arch/amiga/amiga.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								arch/amiga/amiga.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| import "lib/common.proto"; | ||||
|  | ||||
| message AmigaDecoderProto {} | ||||
|  | ||||
| message AmigaEncoderProto { | ||||
| 	optional double clock_rate_us = 1 | ||||
| 		[default=2.00, (help)="Encoded data clock rate."]; | ||||
| 	optional double post_index_gap_ms = 2 | ||||
| 		[default=0.5, (help)="Post-index gap before first sector header."]; | ||||
| } | ||||
|  | ||||
							
								
								
									
										85
									
								
								arch/amiga/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								arch/amiga/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,85 @@ | ||||
| #include "globals.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "protocol.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "amiga.h" | ||||
| #include "bytes.h" | ||||
| #include "fmt/format.h" | ||||
| #include "lib/decoders/decoders.pb.h" | ||||
| #include <string.h> | ||||
| #include <algorithm> | ||||
|  | ||||
| /*  | ||||
|  * Amiga disks use MFM but it's not quite the same as IBM MFM. They only use | ||||
|  * a single type of record with a different marker byte. | ||||
|  *  | ||||
|  * See the big comment in the IBM MFM decoder for the gruesome details of how | ||||
|  * MFM works. | ||||
|  */ | ||||
|           | ||||
| static const FluxPattern SECTOR_PATTERN(48, AMIGA_SECTOR_RECORD); | ||||
|  | ||||
| class AmigaDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	AmigaDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config), | ||||
| 		_config(config.amiga()) | ||||
| 	{} | ||||
|  | ||||
|     RecordType advanceToNextRecord() override | ||||
| 	{ | ||||
| 		_sector->clock = _fmr->seekToPattern(SECTOR_PATTERN); | ||||
| 		if (_fmr->eof() || !_sector->clock) | ||||
| 			return UNKNOWN_RECORD; | ||||
| 		return SECTOR_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() override | ||||
| 	{ | ||||
| 		const auto& rawbits = readRawBits(AMIGA_RECORD_SIZE*16); | ||||
| 		if (rawbits.size() < (AMIGA_RECORD_SIZE*16)) | ||||
| 			return; | ||||
| 		const auto& rawbytes = toBytes(rawbits).slice(0, AMIGA_RECORD_SIZE*2); | ||||
| 		const auto& bytes = decodeFmMfm(rawbits).slice(0, AMIGA_RECORD_SIZE); | ||||
|  | ||||
| 		const uint8_t* ptr = bytes.begin() + 3; | ||||
|  | ||||
| 		Bytes header = amigaDeinterleave(ptr, 4); | ||||
| 		Bytes recoveryinfo = amigaDeinterleave(ptr, 16); | ||||
|  | ||||
| 		_sector->logicalTrack = header[1] >> 1; | ||||
| 		_sector->logicalSide = header[1] & 1; | ||||
| 		_sector->logicalSector = header[2]; | ||||
|  | ||||
| 		uint32_t wantedheaderchecksum = amigaDeinterleave(ptr, 4).reader().read_be32(); | ||||
| 		uint32_t gotheaderchecksum = amigaChecksum(rawbytes.slice(6, 40)); | ||||
| 		if (gotheaderchecksum != wantedheaderchecksum) | ||||
| 			return; | ||||
|  | ||||
| 		uint32_t wanteddatachecksum = amigaDeinterleave(ptr, 4).reader().read_be32(); | ||||
| 		uint32_t gotdatachecksum = amigaChecksum(rawbytes.slice(62, 1024)); | ||||
|  | ||||
| 		Bytes data; | ||||
| 		data.writer().append(amigaDeinterleave(ptr, 512)).append(recoveryinfo); | ||||
| 		_sector->data = data; | ||||
| 		_sector->status = (gotdatachecksum == wanteddatachecksum) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
|  | ||||
| 	std::set<unsigned> requiredSectors(unsigned cylinder, unsigned head) const override | ||||
| 	{ | ||||
| 		static std::set<unsigned> sectors = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const AmigaDecoderProto& _config; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createAmigaDecoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new AmigaDecoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										162
									
								
								arch/amiga/encoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										162
									
								
								arch/amiga/encoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,162 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "amiga.h" | ||||
| #include "crc.h" | ||||
| #include "writer.h" | ||||
| #include "image.h" | ||||
| #include "arch/amiga/amiga.pb.h" | ||||
| #include "lib/encoders/encoders.pb.h" | ||||
|  | ||||
| static bool lastBit; | ||||
|  | ||||
| static int charToInt(char c) | ||||
| { | ||||
| 	if (isdigit(c)) | ||||
| 		return c - '0'; | ||||
| 	return 10 + tolower(c) - 'a'; | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, const std::vector<bool>& src) | ||||
| { | ||||
| 	for (bool bit : src) | ||||
| 	{ | ||||
| 		if (cursor < bits.size()) | ||||
| 			lastBit = bits[cursor++] = bit; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, uint64_t data, int width) | ||||
| { | ||||
| 	cursor += width; | ||||
| 	lastBit = data & 1; | ||||
| 	for (int i=0; i<width; i++) | ||||
| 	{ | ||||
| 		unsigned pos = cursor - i - 1; | ||||
| 		if (pos < bits.size()) | ||||
| 			bits[pos] = data & 1; | ||||
| 		data >>= 1; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, const Bytes& bytes) | ||||
| { | ||||
| 	ByteReader br(bytes); | ||||
| 	BitReader bitr(br); | ||||
|  | ||||
| 	while (!bitr.eof()) | ||||
| 	{ | ||||
| 		if (cursor < bits.size()) | ||||
| 			bits[cursor++] = bitr.get(); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void write_sector(std::vector<bool>& bits, unsigned& cursor, const std::shared_ptr<Sector>& sector) | ||||
| { | ||||
| 	if ((sector->data.size() != 512) && (sector->data.size() != 528)) | ||||
| 		Error() << "unsupported sector size --- you must pick 512 or 528"; | ||||
|  | ||||
| 	uint32_t checksum = 0; | ||||
|  | ||||
| 	auto write_interleaved_bytes = [&](const Bytes& bytes) | ||||
| 	{ | ||||
| 		Bytes interleaved = amigaInterleave(bytes); | ||||
| 		Bytes mfm = encodeMfm(interleaved, lastBit); | ||||
| 		checksum ^= amigaChecksum(mfm); | ||||
| 		checksum &= 0x55555555; | ||||
| 		write_bits(bits, cursor, mfm); | ||||
| 	}; | ||||
|  | ||||
| 	auto write_interleaved_word = [&](uint32_t word) | ||||
| 	{ | ||||
| 		Bytes b(4); | ||||
| 		b.writer().write_be32(word); | ||||
| 		write_interleaved_bytes(b); | ||||
| 	}; | ||||
|  | ||||
|     write_bits(bits, cursor, AMIGA_SECTOR_RECORD, 6*8); | ||||
|  | ||||
| 	checksum = 0; | ||||
| 	Bytes header =  | ||||
| 		{ | ||||
| 			0xff, /* Amiga 1.0 format byte */ | ||||
| 			(uint8_t) ((sector->logicalTrack<<1) | sector->logicalSide), | ||||
| 			(uint8_t) sector->logicalSector, | ||||
| 			(uint8_t) (AMIGA_SECTORS_PER_TRACK - sector->logicalSector) | ||||
| 		}; | ||||
| 	write_interleaved_bytes(header); | ||||
| 	Bytes recoveryInfo(16); | ||||
| 	if (sector->data.size() == 528) | ||||
| 		recoveryInfo = sector->data.slice(512, 16); | ||||
| 	write_interleaved_bytes(recoveryInfo); | ||||
| 	write_interleaved_word(checksum); | ||||
|  | ||||
| 	Bytes data = sector->data.slice(0, 512); | ||||
| 	write_interleaved_word(amigaChecksum(encodeMfm(amigaInterleave(data), lastBit))); | ||||
| 	write_interleaved_bytes(data); | ||||
| } | ||||
|  | ||||
| class AmigaEncoder : public AbstractEncoder | ||||
| { | ||||
| public: | ||||
| 	AmigaEncoder(const EncoderProto& config): | ||||
| 		AbstractEncoder(config), | ||||
| 		_config(config.amiga()) {} | ||||
|  | ||||
| public: | ||||
| 	std::vector<std::shared_ptr<Sector>> collectSectors(int physicalTrack, int physicalSide, const Image& image) override | ||||
| 	{ | ||||
| 		std::vector<std::shared_ptr<Sector>> sectors; | ||||
|  | ||||
| 		if ((physicalTrack >= 0) && (physicalTrack < AMIGA_TRACKS_PER_DISK)) | ||||
| 		{ | ||||
| 			for (int sectorId=0; sectorId<AMIGA_SECTORS_PER_TRACK; sectorId++) | ||||
| 			{ | ||||
| 				const auto& sector = image.get(physicalTrack, physicalSide, sectorId); | ||||
| 				if (sector) | ||||
| 					sectors.push_back(sector); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
|     std::unique_ptr<Fluxmap> encode(int physicalTrack, int physicalSide, | ||||
| 			const std::vector<std::shared_ptr<Sector>>& sectors, const Image& image) override | ||||
| 	{ | ||||
| 		if ((physicalTrack < 0) || (physicalTrack >= AMIGA_TRACKS_PER_DISK)) | ||||
| 			return std::unique_ptr<Fluxmap>(); | ||||
|  | ||||
| 		int bitsPerRevolution = 200000.0 / _config.clock_rate_us(); | ||||
| 		std::vector<bool> bits(bitsPerRevolution); | ||||
| 		unsigned cursor = 0; | ||||
|  | ||||
| 		fillBitmapTo(bits, cursor, _config.post_index_gap_ms() * 1000 / _config.clock_rate_us(), { true, false }); | ||||
| 		lastBit = false; | ||||
|  | ||||
| 		for (int sectorId=0; sectorId<AMIGA_SECTORS_PER_TRACK; sectorId++) | ||||
| 		{ | ||||
| 			const auto& sectorData = image.get(physicalTrack, physicalSide, sectorId); | ||||
| 			if (sectorData) | ||||
| 				write_sector(bits, cursor, sectorData); | ||||
| 		} | ||||
|  | ||||
| 		if (cursor >= bits.size()) | ||||
| 			Error() << "track data overrun"; | ||||
| 		fillBitmapTo(bits, cursor, bits.size(), { true, false }); | ||||
|  | ||||
| 		std::unique_ptr<Fluxmap> fluxmap(new Fluxmap); | ||||
| 		fluxmap->appendBits(bits, _config.clock_rate_us()*1e3); | ||||
| 		return fluxmap; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const AmigaEncoderProto& _config; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractEncoder> createAmigaEncoder(const EncoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractEncoder>(new AmigaEncoder(config)); | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										13
									
								
								arch/apple2/apple2.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								arch/apple2/apple2.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| #ifndef APPLE2_H | ||||
| #define APPLE2_H | ||||
|  | ||||
| #define APPLE2_SECTOR_RECORD   0xd5aa96 | ||||
| #define APPLE2_DATA_RECORD     0xd5aaad | ||||
|  | ||||
| #define APPLE2_SECTOR_LENGTH   256 | ||||
| #define APPLE2_ENCODED_SECTOR_LENGTH 342 | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createApple2Decoder(const DecoderProto& config); | ||||
|  | ||||
| #endif | ||||
|  | ||||
							
								
								
									
										4
									
								
								arch/apple2/apple2.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								arch/apple2/apple2.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| message Apple2DecoderProto {} | ||||
|  | ||||
| @@ -2,7 +2,6 @@ | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "protocol.h" | ||||
| #include "record.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "apple2.h" | ||||
| @@ -25,7 +24,7 @@ static int decode_data_gcr(uint8_t gcr) | ||||
| 		#undef GCR_ENTRY | ||||
|     } | ||||
|     return -1; | ||||
| }; | ||||
| } | ||||
| 
 | ||||
| /* This is extremely inspired by the MESS implementation, written by Nathan Woods
 | ||||
|  * and R. Belmont: https://github.com/mamedev/mame/blob/7914a6083a3b3a8c243ae6c3b8cb50b023f21e0e/src/lib/formats/ap2_dsk.cpp
 | ||||
| @@ -60,53 +59,68 @@ static Bytes decode_crazy_data(const uint8_t* inp, Sector::Status& status) | ||||
|     return output; | ||||
| } | ||||
| 
 | ||||
| uint8_t combine(uint16_t word) | ||||
| static uint8_t combine(uint16_t word) | ||||
| { | ||||
|     return word & (word >> 7); | ||||
| } | ||||
| 
 | ||||
| AbstractDecoder::RecordType Apple2Decoder::advanceToNextRecord() | ||||
| class Apple2Decoder : public AbstractDecoder | ||||
| { | ||||
| 	const FluxMatcher* matcher = nullptr; | ||||
| 	_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
| 	if (matcher == &SECTOR_RECORD_PATTERN) | ||||
| 		return RecordType::SECTOR_RECORD; | ||||
| 	if (matcher == &DATA_RECORD_PATTERN) | ||||
| 		return RecordType::DATA_RECORD; | ||||
| 	return RecordType::UNKNOWN_RECORD; | ||||
| public: | ||||
| 	Apple2Decoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
| 
 | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_RECORD_PATTERN) | ||||
| 			return RecordType::SECTOR_RECORD; | ||||
| 		if (matcher == &DATA_RECORD_PATTERN) | ||||
| 			return RecordType::DATA_RECORD; | ||||
| 		return RecordType::UNKNOWN_RECORD; | ||||
| 	} | ||||
| 
 | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		/* Skip ID (as we know it's a APPLE2_SECTOR_RECORD). */ | ||||
| 		readRawBits(24); | ||||
| 
 | ||||
| 		/* Read header. */ | ||||
| 
 | ||||
| 		auto header = toBytes(readRawBits(8*8)).slice(0, 8); | ||||
| 		ByteReader br(header); | ||||
| 
 | ||||
| 		uint8_t volume = combine(br.read_be16()); | ||||
| 		_sector->logicalTrack = combine(br.read_be16()); | ||||
| 		_sector->logicalSector = combine(br.read_be16()); | ||||
| 		uint8_t checksum = combine(br.read_be16()); | ||||
| 		if (checksum == (volume ^ _sector->logicalTrack ^ _sector->logicalSector)) | ||||
| 			_sector->status = Sector::DATA_MISSING; /* unintuitive but correct */ | ||||
| 	} | ||||
| 
 | ||||
|     void decodeDataRecord() | ||||
| 	{ | ||||
| 		/* Check ID. */ | ||||
| 
 | ||||
| 		Bytes bytes = toBytes(readRawBits(3*8)).slice(0, 3); | ||||
| 		if (bytes.reader().read_be24() != APPLE2_DATA_RECORD) | ||||
| 			return; | ||||
| 
 | ||||
| 		/* Read and decode data. */ | ||||
| 
 | ||||
| 		unsigned recordLength = APPLE2_ENCODED_SECTOR_LENGTH + 2; | ||||
| 		bytes = toBytes(readRawBits(recordLength*8)).slice(0, recordLength); | ||||
| 
 | ||||
| 		_sector->status = Sector::BAD_CHECKSUM; | ||||
| 		_sector->data = decode_crazy_data(&bytes[0], _sector->status); | ||||
| 	} | ||||
| }; | ||||
| 
 | ||||
| std::unique_ptr<AbstractDecoder> createApple2Decoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new Apple2Decoder(config)); | ||||
| } | ||||
| 
 | ||||
| void Apple2Decoder::decodeSectorRecord() | ||||
| { | ||||
|     /* Skip ID (as we know it's a APPLE2_SECTOR_RECORD). */ | ||||
|     readRawBits(24); | ||||
| 
 | ||||
|     /* Read header. */ | ||||
| 
 | ||||
|     auto header = toBytes(readRawBits(8*8)).slice(0, 8); | ||||
|     ByteReader br(header); | ||||
| 
 | ||||
|     uint8_t volume = combine(br.read_be16()); | ||||
|     _sector->logicalTrack = combine(br.read_be16()); | ||||
|     _sector->logicalSector = combine(br.read_be16()); | ||||
|     uint8_t checksum = combine(br.read_be16()); | ||||
|     if (checksum == (volume ^ _sector->logicalTrack ^ _sector->logicalSector)) | ||||
|         _sector->status = Sector::DATA_MISSING; /* unintuitive but correct */ | ||||
| } | ||||
| 
 | ||||
| void Apple2Decoder::decodeDataRecord() | ||||
| { | ||||
|     /* Check ID. */ | ||||
| 
 | ||||
|     Bytes bytes = toBytes(readRawBits(3*8)).slice(0, 3); | ||||
|     if (bytes.reader().read_be24() != APPLE2_DATA_RECORD) | ||||
|         return; | ||||
| 
 | ||||
|     /* Read and decode data. */ | ||||
| 
 | ||||
|     unsigned recordLength = APPLE2_ENCODED_SECTOR_LENGTH + 2; | ||||
|     bytes = toBytes(readRawBits(recordLength*8)).slice(0, recordLength); | ||||
| 
 | ||||
|     _sector->status = Sector::BAD_CHECKSUM; | ||||
|     _sector->data = decode_crazy_data(&bytes[0], _sector->status); | ||||
| } | ||||
							
								
								
									
										19
									
								
								arch/brother/brother.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										19
									
								
								arch/brother/brother.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,19 @@ | ||||
| #ifndef BROTHER_H | ||||
| #define BROTHER_H | ||||
|  | ||||
| /* Brother word processor format (or at least, one of them) */ | ||||
|  | ||||
| #define BROTHER_SECTOR_RECORD            0xFFFFFD57 | ||||
| #define BROTHER_DATA_RECORD              0xFFFFFDDB | ||||
| #define BROTHER_DATA_RECORD_PAYLOAD      256 | ||||
| #define BROTHER_DATA_RECORD_CHECKSUM     3 | ||||
| #define BROTHER_DATA_RECORD_ENCODED_SIZE 415 | ||||
|  | ||||
| #define BROTHER_TRACKS_PER_240KB_DISK    78 | ||||
| #define BROTHER_TRACKS_PER_120KB_DISK    39 | ||||
| #define BROTHER_SECTORS_PER_TRACK        12 | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createBrotherDecoder(const DecoderProto& config); | ||||
| extern std::unique_ptr<AbstractEncoder> createBrotherEncoder(const EncoderProto& config); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										20
									
								
								arch/brother/brother.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										20
									
								
								arch/brother/brother.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,20 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| message BrotherDecoderProto {} | ||||
|  | ||||
| enum BrotherFormat { | ||||
| 	BROTHER240 = 0; | ||||
| 	BROTHER120 = 1; | ||||
| }; | ||||
|  | ||||
| message BrotherEncoderProto { | ||||
| 	optional double clock_rate_us = 1 [default = 3.83]; | ||||
| 	optional double post_index_gap_ms = 2 [default = 1.0]; | ||||
| 	optional double sector_spacing_ms = 3 [default = 16.2]; | ||||
| 	optional double post_header_spacing_ms = 4 [default = 0.69]; | ||||
| 	optional string sector_skew = 5 [default = "05a3816b4927"]; | ||||
|  | ||||
| 	optional BrotherFormat format = 6 [default = BROTHER240]; | ||||
| 	optional int32 bias = 7 [default = 0]; | ||||
| } | ||||
|  | ||||
							
								
								
									
										123
									
								
								arch/brother/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										123
									
								
								arch/brother/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,123 @@ | ||||
| #include "globals.h" | ||||
| #include "sql.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "brother.h" | ||||
| #include "sector.h" | ||||
| #include "bytes.h" | ||||
| #include "crc.h" | ||||
| #include <ctype.h> | ||||
|  | ||||
| const FluxPattern SECTOR_RECORD_PATTERN(32, BROTHER_SECTOR_RECORD); | ||||
| const FluxPattern DATA_RECORD_PATTERN(32, BROTHER_DATA_RECORD); | ||||
| const FluxMatchers ANY_RECORD_PATTERN({ &SECTOR_RECORD_PATTERN, &DATA_RECORD_PATTERN }); | ||||
|  | ||||
| static std::vector<uint8_t> outputbuffer; | ||||
|  | ||||
| /* | ||||
|  * Brother disks have this very very non-IBM system where sector header records | ||||
|  * and data records use two different kinds of GCR: sector headers are 8-in-16 | ||||
|  * (but the encodable values range from 0 to 77ish only) and data headers are | ||||
|  * 5-in-8. In addition, there's a non-encoded 10-bit ID word at the beginning | ||||
|  * of each record, as well as a string of 53 1s introducing them. That does at | ||||
|  * least make them easy to find. | ||||
|  * | ||||
|  * Disk formats vary from machine to machine, but mine uses 78 tracks. Track 0 | ||||
|  * is erased but not formatted.  Track alignment is extremely dubious and | ||||
|  * Brother track 0 shows up on my machine at track 2. | ||||
|  */ | ||||
|  | ||||
| static int decode_data_gcr(uint8_t gcr) | ||||
| { | ||||
|     switch (gcr) | ||||
|     { | ||||
| 		#define GCR_ENTRY(gcr, data) \ | ||||
| 			case gcr: return data; | ||||
| 		#include "data_gcr.h" | ||||
| 		#undef GCR_ENTRY | ||||
|     } | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| static int decode_header_gcr(uint16_t word) | ||||
| { | ||||
| 	switch (word) | ||||
| 	{ | ||||
| 		#define GCR_ENTRY(gcr, data) \ | ||||
| 			case gcr: return data; | ||||
| 		#include "header_gcr.h" | ||||
| 		#undef GCR_ENTRY | ||||
| 	}                        | ||||
| 	return -1;              | ||||
| } | ||||
|  | ||||
| class BrotherDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
|     BrotherDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
|  | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_RECORD_PATTERN) | ||||
| 			return RecordType::SECTOR_RECORD; | ||||
| 		if (matcher == &DATA_RECORD_PATTERN) | ||||
| 			return RecordType::DATA_RECORD; | ||||
| 		return RecordType::UNKNOWN_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		readRawBits(32); | ||||
| 		const auto& rawbits = readRawBits(32); | ||||
| 		const auto& bytes = toBytes(rawbits).slice(0, 4); | ||||
|  | ||||
| 		ByteReader br(bytes); | ||||
| 		_sector->logicalTrack = decode_header_gcr(br.read_be16()); | ||||
| 		_sector->logicalSector = decode_header_gcr(br.read_be16()); | ||||
|  | ||||
| 		/* Sanity check the values read; there's no header checksum and | ||||
| 			* occasionally we get garbage due to bit errors. */ | ||||
| 		if (_sector->logicalSector > 11) | ||||
| 			return; | ||||
| 		if (_sector->logicalTrack > 79) | ||||
| 			return; | ||||
|  | ||||
| 		_sector->status = Sector::DATA_MISSING; | ||||
| 	} | ||||
| 	 | ||||
|     void decodeDataRecord() | ||||
| 	{ | ||||
| 		readRawBits(32); | ||||
|  | ||||
| 		const auto& rawbits = readRawBits(BROTHER_DATA_RECORD_ENCODED_SIZE*8); | ||||
| 		const auto& rawbytes = toBytes(rawbits).slice(0, BROTHER_DATA_RECORD_ENCODED_SIZE); | ||||
|  | ||||
| 		Bytes bytes; | ||||
| 		ByteWriter bw(bytes); | ||||
| 		BitWriter bitw(bw); | ||||
| 		for (uint8_t b : rawbytes) | ||||
| 		{ | ||||
| 			uint32_t nibble = decode_data_gcr(b); | ||||
| 			bitw.push(nibble, 5); | ||||
| 		} | ||||
| 		bitw.flush(); | ||||
|  | ||||
| 		_sector->data = bytes.slice(0, BROTHER_DATA_RECORD_PAYLOAD); | ||||
| 		uint32_t realCrc = crcbrother(_sector->data); | ||||
| 		uint32_t wantCrc = bytes.reader().seek(BROTHER_DATA_RECORD_PAYLOAD).read_be24(); | ||||
| 		_sector->status = (realCrc == wantCrc) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createBrotherDecoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new BrotherDecoder(config)); | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										212
									
								
								arch/brother/encoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										212
									
								
								arch/brother/encoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,212 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "brother.h" | ||||
| #include "crc.h" | ||||
| #include "writer.h" | ||||
| #include "image.h" | ||||
| #include "arch/brother/brother.pb.h" | ||||
| #include "lib/encoders/encoders.pb.h" | ||||
|  | ||||
| static int encode_header_gcr(uint16_t word) | ||||
| { | ||||
| 	switch (word) | ||||
| 	{ | ||||
| 		#define GCR_ENTRY(gcr, data) \ | ||||
| 			case data: return gcr; | ||||
| 		#include "header_gcr.h" | ||||
| 		#undef GCR_ENTRY | ||||
| 	}                        | ||||
| 	return -1;              | ||||
| } | ||||
|  | ||||
| static int encode_data_gcr(uint8_t data) | ||||
| { | ||||
| 	switch (data) | ||||
| 	{ | ||||
| 		#define GCR_ENTRY(gcr, data) \ | ||||
| 			case data: return gcr; | ||||
| 		#include "data_gcr.h" | ||||
| 		#undef GCR_ENTRY | ||||
| 	}                        | ||||
| 	return -1;              | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, uint32_t data, int width) | ||||
| { | ||||
| 	cursor += width; | ||||
| 	for (int i=0; i<width; i++) | ||||
| 	{ | ||||
| 		unsigned pos = cursor - i - 1; | ||||
| 		if (pos < bits.size()) | ||||
| 			bits[pos] = data & 1; | ||||
| 		data >>= 1; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void write_sector_header(std::vector<bool>& bits, unsigned& cursor, | ||||
| 		int track, int sector) | ||||
| { | ||||
| 	write_bits(bits, cursor, 0xffffffff, 31); | ||||
| 	write_bits(bits, cursor, BROTHER_SECTOR_RECORD, 32); | ||||
| 	write_bits(bits, cursor, encode_header_gcr(track), 16); | ||||
| 	write_bits(bits, cursor, encode_header_gcr(sector), 16); | ||||
| 	write_bits(bits, cursor, encode_header_gcr(0x2f), 16); | ||||
| } | ||||
|  | ||||
| static void write_sector_data(std::vector<bool>& bits, unsigned& cursor, const Bytes& data) | ||||
| { | ||||
| 	write_bits(bits, cursor, 0xffffffff, 32); | ||||
| 	write_bits(bits, cursor, BROTHER_DATA_RECORD, 32); | ||||
|  | ||||
| 	uint16_t fifo = 0; | ||||
| 	int width = 0; | ||||
|  | ||||
| 	if (data.size() != BROTHER_DATA_RECORD_PAYLOAD) | ||||
| 		Error() << "unsupported sector size"; | ||||
|  | ||||
| 	auto write_byte = [&](uint8_t byte) | ||||
| 	{ | ||||
| 		fifo |= (byte << (8 - width)); | ||||
| 		width += 8; | ||||
|  | ||||
| 		while (width >= 5) | ||||
| 		{ | ||||
| 			uint8_t quintet = fifo >> 11; | ||||
| 			fifo <<= 5; | ||||
| 			width -= 5; | ||||
|  | ||||
| 			write_bits(bits, cursor, encode_data_gcr(quintet), 8); | ||||
| 		} | ||||
| 	}; | ||||
|  | ||||
| 	for (uint8_t byte : data) | ||||
| 		write_byte(byte); | ||||
|  | ||||
| 	uint32_t realCrc = crcbrother(data); | ||||
| 	write_byte(realCrc>>16); | ||||
| 	write_byte(realCrc>>8); | ||||
| 	write_byte(realCrc); | ||||
| 	write_byte(0x58); /* magic */ | ||||
|     write_byte(0xd4); | ||||
|     while (width != 0) | ||||
|         write_byte(0); | ||||
| } | ||||
|  | ||||
| static int charToInt(char c) | ||||
| { | ||||
| 	if (isdigit(c)) | ||||
| 		return c - '0'; | ||||
| 	return 10 + tolower(c) - 'a'; | ||||
| } | ||||
|  | ||||
| class BrotherEncoder : public AbstractEncoder | ||||
| { | ||||
| public: | ||||
| 	BrotherEncoder(const EncoderProto& config): | ||||
| 		AbstractEncoder(config), | ||||
| 		_config(config.brother()) | ||||
| 	{} | ||||
|  | ||||
| public: | ||||
| 	std::vector<std::shared_ptr<Sector>> collectSectors(int physicalTrack, int physicalSide, const Image& image) override | ||||
| 	{ | ||||
| 		std::vector<std::shared_ptr<Sector>> sectors; | ||||
|  | ||||
| 		int logicalTrack; | ||||
| 		if (physicalSide != 0) | ||||
| 			return sectors; | ||||
| 		physicalTrack -= _config.bias(); | ||||
| 		switch (_config.format()) | ||||
| 		{ | ||||
| 			case BROTHER120: | ||||
| 				if ((physicalTrack < 0) || (physicalTrack >= (BROTHER_TRACKS_PER_120KB_DISK*2)) | ||||
| 						|| (physicalTrack & 1)) | ||||
| 					return sectors; | ||||
| 				logicalTrack = physicalTrack/2; | ||||
| 				break; | ||||
|  | ||||
| 			case BROTHER240: | ||||
| 				if ((physicalTrack < 0) || (physicalTrack >= BROTHER_TRACKS_PER_240KB_DISK)) | ||||
| 					return sectors; | ||||
| 				logicalTrack = physicalTrack; | ||||
| 				break; | ||||
| 		} | ||||
|  | ||||
|         for (int sectorId=0; sectorId<BROTHER_SECTORS_PER_TRACK; sectorId++) | ||||
|         { | ||||
|             const auto& sector = image.get(logicalTrack, 0, sectorId); | ||||
|             if (sector) | ||||
|                 sectors.push_back(sector); | ||||
| 		} | ||||
|  | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
|     std::unique_ptr<Fluxmap> encode(int physicalTrack, int physicalSide, | ||||
| 			const std::vector<std::shared_ptr<Sector>>& sectors, const Image& image) override | ||||
| 	{ | ||||
| 		int logicalTrack; | ||||
| 		if (physicalSide != 0) | ||||
| 			return std::unique_ptr<Fluxmap>(); | ||||
| 		physicalTrack -= _config.bias(); | ||||
| 		switch (_config.format()) | ||||
| 		{ | ||||
| 			case BROTHER120: | ||||
| 				if ((physicalTrack < 0) || (physicalTrack >= (BROTHER_TRACKS_PER_120KB_DISK*2)) | ||||
| 						|| (physicalTrack & 1)) | ||||
| 					return std::unique_ptr<Fluxmap>(); | ||||
| 				logicalTrack = physicalTrack/2; | ||||
| 				break; | ||||
|  | ||||
| 			case BROTHER240: | ||||
| 				if ((physicalTrack < 0) || (physicalTrack >= BROTHER_TRACKS_PER_240KB_DISK)) | ||||
| 					return std::unique_ptr<Fluxmap>(); | ||||
| 				logicalTrack = physicalTrack; | ||||
| 				break; | ||||
| 		} | ||||
|  | ||||
| 		int bitsPerRevolution = 200000.0 / _config.clock_rate_us(); | ||||
| 		const std::string& skew = _config.sector_skew(); | ||||
| 		std::vector<bool> bits(bitsPerRevolution); | ||||
| 		unsigned cursor = 0; | ||||
|  | ||||
| 		for (int sectorCount=0; sectorCount<BROTHER_SECTORS_PER_TRACK; sectorCount++) | ||||
| 		{ | ||||
| 			int sectorId = charToInt(skew.at(sectorCount)); | ||||
| 			double headerMs = _config.post_index_gap_ms() + sectorCount*_config.sector_spacing_ms(); | ||||
| 			unsigned headerCursor = headerMs*1e3 / _config.clock_rate_us(); | ||||
| 			double dataMs = headerMs + _config.post_header_spacing_ms(); | ||||
| 			unsigned dataCursor = dataMs*1e3 / _config.clock_rate_us(); | ||||
|  | ||||
| 			const auto& sectorData = image.get(logicalTrack, 0, sectorId); | ||||
|  | ||||
| 			fillBitmapTo(bits, cursor, headerCursor, { true, false }); | ||||
| 			write_sector_header(bits, cursor, logicalTrack, sectorId); | ||||
| 			fillBitmapTo(bits, cursor, dataCursor, { true, false }); | ||||
| 			write_sector_data(bits, cursor, sectorData->data); | ||||
| 		} | ||||
|  | ||||
| 		if (cursor >= bits.size()) | ||||
| 			Error() << "track data overrun"; | ||||
| 		fillBitmapTo(bits, cursor, bits.size(), { true, false }); | ||||
|  | ||||
| 		// The pre-index gap is not normally reported. | ||||
| 		// std::cerr << "pre-index gap " << 200.0 - (double)cursor*clockRateUs/1e3 << std::endl; | ||||
| 		 | ||||
| 		std::unique_ptr<Fluxmap> fluxmap(new Fluxmap); | ||||
| 		fluxmap->appendBits(bits, _config.clock_rate_us()*1e3); | ||||
| 		return fluxmap; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const BrotherEncoderProto& _config; | ||||
|  | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractEncoder> createBrotherEncoder(const EncoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractEncoder>(new BrotherEncoder(config)); | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										33
									
								
								arch/c64/c64.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										33
									
								
								arch/c64/c64.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,33 @@ | ||||
| #ifndef C64_H | ||||
| #define C64_H | ||||
|  | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
|  | ||||
| #define C64_SECTOR_RECORD    0xffd49 | ||||
| #define C64_DATA_RECORD      0xffd57 | ||||
| #define C64_SECTOR_LENGTH    256 | ||||
|  | ||||
| /* Source: http://www.unusedino.de/ec64/technical/formats/g64.html  | ||||
|    1. Header sync       FF FF FF FF FF (40 'on' bits, not GCR) | ||||
|    2. Header info       52 54 B5 29 4B 7A 5E 95 55 55 (10 GCR bytes) | ||||
|    3. Header gap        55 55 55 55 55 55 55 55 55 (9 bytes, never read) | ||||
|    4. Data sync         FF FF FF FF FF (40 'on' bits, not GCR) | ||||
|    5. Data block        55...4A (325 GCR bytes) | ||||
|    6. Inter-sector gap  55 55 55 55...55 55 (4 to 12 bytes, never read) | ||||
|    1. Header sync       (SYNC for the next sector) | ||||
| */ | ||||
| #define C64_HEADER_DATA_SYNC        0xFF | ||||
| #define C64_HEADER_BLOCK_ID         0x08 | ||||
| #define C64_DATA_BLOCK_ID           0x07 | ||||
| #define C64_HEADER_GAP              0x55 | ||||
| #define C64_INTER_SECTOR_GAP        0x55 | ||||
| #define C64_PADDING                 0x0F | ||||
|  | ||||
| #define C64_TRACKS_PER_DISK         40 | ||||
| #define C64_BAM_TRACK               17 | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createCommodore64Decoder(const DecoderProto& config); | ||||
| extern std::unique_ptr<AbstractEncoder> createCommodore64Encoder(const EncoderProto& config); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										13
									
								
								arch/c64/c64.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								arch/c64/c64.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| import "lib/common.proto"; | ||||
|  | ||||
| message Commodore64DecoderProto {} | ||||
|  | ||||
| message Commodore64EncoderProto { | ||||
| 	optional double post_index_gap_us = 1 [default=0.0, | ||||
| 		(help) = "post-index gap before first sector header."]; | ||||
| 	optional double clock_compensation_factor = 2 [default=1.0, | ||||
| 		(help) = "scale the output clock by this much."]; | ||||
| } | ||||
|  | ||||
							
								
								
									
										106
									
								
								arch/c64/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								arch/c64/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,106 @@ | ||||
| #include "globals.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "protocol.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "c64.h" | ||||
| #include "crc.h" | ||||
| #include "bytes.h" | ||||
| #include "fmt/format.h" | ||||
| #include <string.h> | ||||
| #include <algorithm> | ||||
|  | ||||
| const FluxPattern SECTOR_RECORD_PATTERN(20, C64_SECTOR_RECORD); | ||||
| const FluxPattern DATA_RECORD_PATTERN(20, C64_DATA_RECORD); | ||||
| const FluxMatchers ANY_RECORD_PATTERN({ &SECTOR_RECORD_PATTERN, &DATA_RECORD_PATTERN }); | ||||
|  | ||||
| static int decode_data_gcr(uint8_t gcr) | ||||
| { | ||||
|     switch (gcr) | ||||
|     { | ||||
| 		#define GCR_ENTRY(gcr, data) \ | ||||
| 			case gcr: return data; | ||||
| 		#include "data_gcr.h" | ||||
| 		#undef GCR_ENTRY | ||||
|     } | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| static Bytes decode(const std::vector<bool>& bits) | ||||
| { | ||||
|     Bytes output; | ||||
|     ByteWriter bw(output); | ||||
|     BitWriter bitw(bw); | ||||
|  | ||||
|     auto ii = bits.begin(); | ||||
|     while (ii != bits.end()) | ||||
|     { | ||||
|         uint8_t inputfifo = 0; | ||||
|         for (size_t i=0; i<5; i++) | ||||
|         { | ||||
|             if (ii == bits.end()) | ||||
|                 break; | ||||
|             inputfifo = (inputfifo<<1) | *ii++; | ||||
|         } | ||||
|  | ||||
|         bitw.push(decode_data_gcr(inputfifo), 4); | ||||
|     } | ||||
|     bitw.flush(); | ||||
|  | ||||
|     return output; | ||||
| } | ||||
|  | ||||
| class Commodore64Decoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	Commodore64Decoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
|  | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_RECORD_PATTERN) | ||||
| 			return RecordType::SECTOR_RECORD; | ||||
| 		if (matcher == &DATA_RECORD_PATTERN) | ||||
| 			return RecordType::DATA_RECORD; | ||||
| 		return RecordType::UNKNOWN_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		readRawBits(20); | ||||
|  | ||||
| 		const auto& bits = readRawBits(5*10); | ||||
| 		const auto& bytes = decode(bits).slice(0, 5); | ||||
|  | ||||
| 		uint8_t checksum = bytes[0]; | ||||
| 		_sector->logicalSector = bytes[1]; | ||||
| 		_sector->logicalSide = 0; | ||||
| 		_sector->logicalTrack = bytes[2] - 1; | ||||
| 		if (checksum == xorBytes(bytes.slice(1, 4))) | ||||
| 			_sector->status = Sector::DATA_MISSING; /* unintuitive but correct */ | ||||
| 	} | ||||
|  | ||||
|     void decodeDataRecord() | ||||
| 	{ | ||||
| 		readRawBits(20); | ||||
|  | ||||
| 		const auto& bits = readRawBits(259*10); | ||||
| 		const auto& bytes = decode(bits).slice(0, 259); | ||||
|  | ||||
| 		_sector->data = bytes.slice(0, C64_SECTOR_LENGTH); | ||||
| 		uint8_t gotChecksum = xorBytes(_sector->data); | ||||
| 		uint8_t wantChecksum = bytes[256]; | ||||
| 		_sector->status = (wantChecksum == gotChecksum) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createCommodore64Decoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new Commodore64Decoder(config)); | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										381
									
								
								arch/c64/encoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										381
									
								
								arch/c64/encoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,381 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "c64.h" | ||||
| #include "crc.h" | ||||
| #include "sector.h" | ||||
| #include "writer.h" | ||||
| #include "image.h" | ||||
| #include "fmt/format.h" | ||||
| #include "arch/c64/c64.pb.h" | ||||
| #include "lib/encoders/encoders.pb.h" | ||||
| #include <ctype.h> | ||||
| #include "bytes.h" | ||||
|  | ||||
| static bool lastBit; | ||||
|  | ||||
| static double clockRateUsForTrack(unsigned track) | ||||
| { | ||||
|     /* | ||||
|      * Track   # Sectors/Track Speed Zone  bits/rotation        | ||||
|      * 1 – 17          21          3       61,538.4 | ||||
|      * 18 – 24         19          2       57,142.8 | ||||
|      * 25 – 30         18          1       53,333.4 | ||||
|      * 31 – 35         17          0       50,000.0 | ||||
|     */ | ||||
|     if (track < 17) | ||||
|         return 200000.0/61538.4; | ||||
|     if (track < 24) | ||||
|         return 200000.0/57142.8; | ||||
|     if (track < 30) | ||||
|         return 200000.0/53333.4; | ||||
|     return 200000.0/50000.0; | ||||
|  | ||||
| } | ||||
|  | ||||
| static unsigned sectorsForTrack(unsigned track) | ||||
| { | ||||
|     /* | ||||
|      * Track   Sectors/track   # Sectors   Storage in Bytes | ||||
|      *   -----   -------------   ---------   ---------------- | ||||
|      *    1-17        21            357           7820 | ||||
|      *   18-24        19            133           7170 | ||||
|      *   25-30        18            108           6300 | ||||
|      *   31-40(*)     17             85           6020 | ||||
|      *                              --- | ||||
|      *                              683 (for a 35 track image) | ||||
|      */ | ||||
|     if (track < 17) | ||||
|         return 21; | ||||
|     if (track < 24) | ||||
|         return 19; | ||||
|     if (track < 30) | ||||
|         return 18; | ||||
|     return 17; | ||||
| } | ||||
|  | ||||
| static int encode_data_gcr(uint8_t data) | ||||
| { | ||||
|     switch (data) | ||||
|     { | ||||
|         #define GCR_ENTRY(gcr, data) \ | ||||
|             case data: return gcr; | ||||
|         #include "data_gcr.h" | ||||
|         #undef GCR_ENTRY | ||||
|     } | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, const std::vector<bool>& src) | ||||
| { | ||||
|     for (bool bit : src)  //Range-based for loop | ||||
|     { | ||||
|         if (cursor < bits.size()) | ||||
|             bits[cursor++] = bit; | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, uint64_t data, int width) | ||||
| { | ||||
|     cursor += width; | ||||
|     for (int i=0; i<width; i++) | ||||
|     { | ||||
|         unsigned pos = cursor - i - 1;               | ||||
|         if (pos < bits.size()) | ||||
|             bits[pos] = data & 1; | ||||
|         data >>= 1; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void bindump(std::ostream& stream, std::vector<bool>& buffer) | ||||
| { | ||||
|     size_t pos = 0; | ||||
|  | ||||
|     while ((pos < buffer.size()) and (pos <520)) | ||||
|     { | ||||
|         stream << fmt::format("{:5d} : ", pos); | ||||
|         for (int i=0; i<40; i++) | ||||
|         { | ||||
|             if ((pos+i) < buffer.size()) | ||||
|                 stream << fmt::format("{:01b}", (buffer[pos+i])); | ||||
|             else | ||||
|                 stream << "-- "; | ||||
|             if ((((pos + i + 1) % 8) == 0) and i != 0) | ||||
|                 stream << "  "; | ||||
|  | ||||
|         } | ||||
|         stream << std::endl; | ||||
|         pos += 40; | ||||
|     } | ||||
| } | ||||
| static std::vector<bool> encode_data(uint8_t input) | ||||
| { | ||||
|     /* | ||||
|     * Four 8-bit data bytes are converted to four 10-bit GCR bytes at a time by | ||||
|     * the 1541 DOS.  RAM is only an 8-bit storage device though. This hardware | ||||
|     * limitation prevents a 10-bit GCR byte from being stored in a single | ||||
|     * memory location. Four 10-bit GCR bytes total 40 bits - a number evenly | ||||
|     * divisible by our overriding 8-bit constraint. Commodore sub- divides the | ||||
|     * 40 GCR bits into five 8-bit bytes to solve this dilemma. This explains | ||||
|     * why four 8-bit data bytes are converted to GCR form at a time. The | ||||
|     * following step by step example demonstrates how this bit manipulation is | ||||
|     * performed by the DOS. | ||||
|     * | ||||
|     * STEP 1. Four 8-bit Data Bytes | ||||
|     * $08 $10 $00 $12 | ||||
|     * | ||||
|     * STEP 2. Hexadecimal to Binary Conversion | ||||
|     * 1. Binary Equivalents | ||||
|     * $08       $10         $00         $12 | ||||
|     * 00001000  00010000    00000000    00010010 | ||||
|     * | ||||
|     * STEP 3. Binary to GCR Conversion | ||||
|     * 1. Four 8-bit Data Bytes | ||||
|     * 00001000 00010000 00000000 00010010 | ||||
|     * 2. High and Low Nybbles | ||||
|     * 0000 1000 0001 0000 0000 0000 0001 0010 | ||||
|     * 3. High and Low Nybble GCR Equivalents | ||||
|     * 01010 01001 01011 01010 01010 01010 01011 10010 | ||||
|     * 4. Four 10-bit GCR Bytes | ||||
|     * 0101001001 0101101010 0101001010 0101110010 | ||||
|     * | ||||
|     * STEP 4. 10-bit GCR to 8-bit GCR Conversion | ||||
|     *   1. Concatenate Four 10-bit GCR Bytes | ||||
|     *   0101001001010110101001010010100101110010 | ||||
|     *   2. Five 8-bit Subdivisions | ||||
|     *   01010010 01010110 10100101 00101001 01110010 | ||||
|     * | ||||
|     * STEP 5. Binary to Hexadecimal Conversion | ||||
|     * 1. Hexadecimal Equivalents | ||||
|     *   01010010    01010110    10100101    00101001    01110010 | ||||
|     *   $52         $56         $A5         $29         $72 | ||||
|     * | ||||
|     * STEP 6. Four 8-bit Data Bytes are Recorded as Five 8-bit GCR Bytes | ||||
|     *   $08 $10 $00 $12 | ||||
|     * | ||||
|     * are recorded as | ||||
|     *   $52 $56 $A5 $29 $72 | ||||
|     */ | ||||
|  | ||||
|     std::vector<bool> output(10, false); | ||||
|     uint8_t hi = 0; | ||||
|     uint8_t lo = 0; | ||||
|     uint8_t lo_GCR = 0; | ||||
|     uint8_t hi_GCR = 0; | ||||
|  | ||||
|     //Convert the byte in high and low nibble | ||||
|     lo = input >> 4; //get the lo nibble shift the bits 4 to the right           | ||||
|     hi = input & 15; //get the hi nibble bij masking the lo bits (00001111)      | ||||
|  | ||||
|  | ||||
|     lo_GCR = encode_data_gcr(lo);   //example value: 0000   GCR = 01010 | ||||
|     hi_GCR = encode_data_gcr(hi);   //example value: 1000   GCR = 01001 | ||||
|     //output = [0,1,2,3,4,5,6,7,8,9] | ||||
|     //value  = [0,1,0,1,0,0,1,0,0,1] | ||||
|     //          01010 01001 | ||||
|  | ||||
|     int b = 4; | ||||
|     for (int i = 0; i < 10; i++) | ||||
|     { | ||||
|         if (i < 5) //01234 | ||||
|         {       //i = 0 op  | ||||
|             output[4-i] = (lo_GCR & 1); //01010 | ||||
|  | ||||
|             //01010 -> & 00001 -> 00000 output[4] = 0 | ||||
|             //00101 -> & 00001 -> 00001 output[3] = 1 | ||||
|             //00010 -> & 00001 -> 00000 output[2] = 0 | ||||
|             //00001 -> & 00001 -> 00001 output[1] = 1 | ||||
|             //00000 -> & 00001 -> 00000 output[0] = 0 | ||||
|             lo_GCR >>= 1; | ||||
|         } else   | ||||
|         { | ||||
|             output[i+b] = (hi_GCR & 1); //01001 | ||||
|             //01001 -> & 00001 -> 00001 output[9] = 1 | ||||
|             //00100 -> & 00001 -> 00000 output[8] = 0 | ||||
|             //00010 -> & 00001 -> 00000 output[7] = 0 | ||||
|             //00001 -> & 00001 -> 00001 output[6] = 1 | ||||
|             //00000 -> & 00001 -> 00000 output[5] = 0 | ||||
|             hi_GCR >>= 1; | ||||
|             b = b-2; | ||||
|         } | ||||
|     } | ||||
|     return output; | ||||
| } | ||||
|  | ||||
| class Commodore64Encoder : public AbstractEncoder | ||||
| { | ||||
| public: | ||||
| 	Commodore64Encoder(const EncoderProto& config): | ||||
|         AbstractEncoder(config), | ||||
| 		_config(config.c64()) | ||||
| 	{} | ||||
|  | ||||
| public: | ||||
| 	std::vector<std::shared_ptr<Sector>> collectSectors(int physicalTrack, int physicalSide, const Image& image) override | ||||
| 	{ | ||||
| 		std::vector<std::shared_ptr<Sector>> sectors; | ||||
|  | ||||
|         if (physicalSide == 0) | ||||
|         { | ||||
|             int logicalTrack = physicalTrack / 2; | ||||
|             unsigned numSectors = sectorsForTrack(logicalTrack); | ||||
|             for (int sectorId=0; sectorId<numSectors; sectorId++) | ||||
|             { | ||||
|                 const auto& sector = image.get(logicalTrack, 0, sectorId); | ||||
|                 if (sector) | ||||
|                     sectors.push_back(sector); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
|     std::unique_ptr<Fluxmap> encode(int physicalTrack, int physicalSide, | ||||
|             const std::vector<std::shared_ptr<Sector>>& sectors, const Image& image) override | ||||
|     { | ||||
|         /* The format ID Character # 1 and # 2 are in the .d64 image only present | ||||
|          * in track 18 sector zero which contains the BAM info in byte 162 and 163. | ||||
|          * it is written in every header of every sector and track. headers are not | ||||
|          * stored in a d64 disk image so we have to get it from track 18 which | ||||
|          * contains the BAM. | ||||
|         */ | ||||
|  | ||||
|         if (physicalSide != 0) | ||||
|             return std::unique_ptr<Fluxmap>(); | ||||
|  | ||||
|         const auto& sectorData = image.get(C64_BAM_TRACK*2, 0, 0); //Read de BAM to get the DISK ID bytes | ||||
|         if (sectorData) | ||||
|         { | ||||
|             ByteReader br(sectorData->data); | ||||
|             br.seek(162); //goto position of the first Disk ID Byte | ||||
|             _formatByte1 = br.read_8(); | ||||
|             _formatByte2 = br.read_8(); | ||||
|         } | ||||
|         else | ||||
|             _formatByte1 = _formatByte2 = 0; | ||||
|          | ||||
|         int logicalTrack = physicalTrack / 2; | ||||
|         double clockRateUs = clockRateUsForTrack(logicalTrack) * _config.clock_compensation_factor(); | ||||
|  | ||||
|         int bitsPerRevolution = 200000.0 / clockRateUs; | ||||
|  | ||||
|         std::vector<bool> bits(bitsPerRevolution); | ||||
|         unsigned cursor = 0; | ||||
|  | ||||
|         fillBitmapTo(bits, cursor, _config.post_index_gap_us() / clockRateUs, { true, false }); | ||||
|         lastBit = false; | ||||
|  | ||||
|         for (const auto& sector : sectors) | ||||
|             writeSector(bits, cursor, sector); | ||||
|  | ||||
|         if (cursor >= bits.size()) | ||||
|             Error() << fmt::format("track data overrun by {} bits", cursor - bits.size()); | ||||
|         fillBitmapTo(bits, cursor, bits.size(), { true, false }); | ||||
|  | ||||
|         std::unique_ptr<Fluxmap> fluxmap(new Fluxmap); | ||||
|         fluxmap->appendBits(bits, clockRateUs*1e3); | ||||
|         return fluxmap; | ||||
|     } | ||||
|  | ||||
| private: | ||||
| 	void writeSector(std::vector<bool>& bits, unsigned& cursor, const std::shared_ptr<Sector>& sector) const | ||||
|     { | ||||
|         /* Source: http://www.unusedino.de/ec64/technical/formats/g64.html  | ||||
|          * 1. Header sync       FF FF FF FF FF (40 'on' bits, not GCR) | ||||
|          * 2. Header info       52 54 B5 29 4B 7A 5E 95 55 55 (10 GCR bytes) | ||||
|          * 3. Header gap        55 55 55 55 55 55 55 55 55 (9 bytes, never read) | ||||
|          * 4. Data sync         FF FF FF FF FF (40 'on' bits, not GCR) | ||||
|          * 5. Data block        55...4A (325 GCR bytes) | ||||
|          * 6. Inter-sector gap  55 55 55 55...55 55 (4 to 12 bytes, never read) | ||||
|          * 1. Header sync       (SYNC for the next sector) | ||||
|         */ | ||||
|         if ((sector->status == Sector::OK) or (sector->status == Sector::BAD_CHECKSUM)) | ||||
|         { | ||||
|             // There is data to encode to disk. | ||||
|             if ((sector->data.size() != C64_SECTOR_LENGTH)) | ||||
|                 Error() << fmt::format("unsupported sector size {} --- you must pick 256", sector->data.size());     | ||||
|  | ||||
|             // 1. Write header Sync (not GCR) | ||||
|             for (int i=0; i<6; i++) | ||||
|                 write_bits(bits, cursor, C64_HEADER_DATA_SYNC, 1*8); /* sync */ | ||||
|  | ||||
|             // 2. Write Header info 10 GCR bytes | ||||
|             /* | ||||
|              * The 10 byte header info (#2) is GCR encoded and must be decoded  to | ||||
|              * it's normal 8 bytes to be understood. Once decoded, its breakdown is | ||||
|              * as follows: | ||||
|              *  | ||||
|              * Byte $00 - header block ID           ($08) | ||||
|              *   01 - header block checksum 16  (EOR of $02-$05) | ||||
|              *   02 - Sector | ||||
|              *   03 - Track | ||||
|              *   04 - Format ID byte #2 | ||||
|              *   05 - Format ID byte #1 | ||||
|              *   06-07 - $0F ("off" bytes) | ||||
|              */ | ||||
|             uint8_t encodedTrack = ((sector->logicalTrack) + 1); // C64 track numbering starts with 1. Fluxengine with 0. | ||||
|             uint8_t encodedSector = sector->logicalSector; | ||||
|             // uint8_t formatByte1 = C64_FORMAT_ID_BYTE1; | ||||
|             // uint8_t formatByte2 = C64_FORMAT_ID_BYTE2; | ||||
|             uint8_t headerChecksum = (encodedTrack ^ encodedSector ^ _formatByte1 ^ _formatByte2); | ||||
|             write_bits(bits, cursor, encode_data(C64_HEADER_BLOCK_ID)); | ||||
|             write_bits(bits, cursor, encode_data(headerChecksum)); | ||||
|             write_bits(bits, cursor, encode_data(encodedSector)); | ||||
|             write_bits(bits, cursor, encode_data(encodedTrack)); | ||||
|             write_bits(bits, cursor, encode_data(_formatByte2)); | ||||
|             write_bits(bits, cursor, encode_data(_formatByte1)); | ||||
|             write_bits(bits, cursor, encode_data(C64_PADDING)); | ||||
|             write_bits(bits, cursor, encode_data(C64_PADDING)); | ||||
|  | ||||
|             // 3. Write header GAP not GCR | ||||
|             for (int i=0; i<9; i++) | ||||
|                 write_bits(bits, cursor, C64_HEADER_GAP, 1*8); /* header gap */ | ||||
|  | ||||
|             // 4. Write Data sync not GCR | ||||
|             for (int i=0; i<6; i++) | ||||
|                 write_bits(bits, cursor, C64_HEADER_DATA_SYNC, 1*8); /* sync */ | ||||
|  | ||||
|             // 5. Write data block 325 GCR bytes | ||||
|             /* | ||||
|              * The 325 byte data block (#5) is GCR encoded and must be  decoded  to  its | ||||
|              * normal 260 bytes to be understood. The data block is made up of the following: | ||||
|              *  | ||||
|              * Byte    $00 - data block ID ($07) | ||||
|              *      01-100 - 256 bytes data | ||||
|              *      101 - data block checksum (EOR of $01-100) | ||||
|              *      102-103 - $00 ("off" bytes, to make the sector size a multiple of 5) | ||||
|              */ | ||||
|  | ||||
|             write_bits(bits, cursor, encode_data(C64_DATA_BLOCK_ID)); | ||||
|             uint8_t dataChecksum = xorBytes(sector->data); | ||||
|             ByteReader br(sector->data); | ||||
|             int i = 0; | ||||
|             for (i = 0; i < C64_SECTOR_LENGTH; i++) | ||||
|             { | ||||
|                 uint8_t val = br.read_8(); | ||||
|                 write_bits(bits, cursor, encode_data(val));      | ||||
|             } | ||||
|             write_bits(bits, cursor, encode_data(dataChecksum)); | ||||
|             write_bits(bits, cursor, encode_data(C64_PADDING)); | ||||
|             write_bits(bits, cursor, encode_data(C64_PADDING)); | ||||
|  | ||||
|             //6. Write inter-sector gap 9 - 12 bytes nor gcr | ||||
|             for (int i=0; i<9; i++) | ||||
|                 write_bits(bits, cursor, C64_INTER_SECTOR_GAP, 1*8); /* sync */ | ||||
|  | ||||
|         } | ||||
|     } | ||||
|          | ||||
| private: | ||||
| 	const Commodore64EncoderProto& _config; | ||||
| 	uint8_t _formatByte1; | ||||
| 	uint8_t _formatByte2; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractEncoder> createCommodore64Encoder(const EncoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractEncoder>(new Commodore64Encoder(config)); | ||||
| } | ||||
|  | ||||
| // vim: sw=4 ts=4 et | ||||
|  | ||||
							
								
								
									
										113
									
								
								arch/f85/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								arch/f85/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | ||||
| #include "globals.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "protocol.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "f85.h" | ||||
| #include "crc.h" | ||||
| #include "bytes.h" | ||||
| #include "fmt/format.h" | ||||
| #include <string.h> | ||||
| #include <algorithm> | ||||
|  | ||||
| const FluxPattern SECTOR_RECORD_PATTERN(24, F85_SECTOR_RECORD); | ||||
| const FluxPattern DATA_RECORD_PATTERN(24, F85_DATA_RECORD); | ||||
| const FluxMatchers ANY_RECORD_PATTERN({ &SECTOR_RECORD_PATTERN, &DATA_RECORD_PATTERN }); | ||||
|  | ||||
| static int decode_data_gcr(uint8_t gcr) | ||||
| { | ||||
|     switch (gcr) | ||||
|     { | ||||
| 		#define GCR_ENTRY(gcr, data) \ | ||||
| 			case gcr: return data; | ||||
| 		#include "data_gcr.h" | ||||
| 		#undef GCR_ENTRY | ||||
|     } | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| static Bytes decode(const std::vector<bool>& bits) | ||||
| { | ||||
|     Bytes output; | ||||
|     ByteWriter bw(output); | ||||
|     BitWriter bitw(bw); | ||||
|  | ||||
|     auto ii = bits.begin(); | ||||
|     while (ii != bits.end()) | ||||
|     { | ||||
|         uint8_t inputfifo = 0; | ||||
|         for (size_t i=0; i<5; i++) | ||||
|         { | ||||
|             if (ii == bits.end()) | ||||
|                 break; | ||||
|             inputfifo = (inputfifo<<1) | *ii++; | ||||
|         } | ||||
|  | ||||
|         bitw.push(decode_data_gcr(inputfifo), 4); | ||||
|     } | ||||
|     bitw.flush(); | ||||
|  | ||||
|     return output; | ||||
| } | ||||
|  | ||||
| class DurangoF85Decoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	DurangoF85Decoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
|  | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_RECORD_PATTERN) | ||||
| 			return RecordType::SECTOR_RECORD; | ||||
| 		if (matcher == &DATA_RECORD_PATTERN) | ||||
| 			return RecordType::DATA_RECORD; | ||||
| 		return RecordType::UNKNOWN_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		/* Skip sync bits and ID byte. */ | ||||
|  | ||||
| 		readRawBits(24); | ||||
|  | ||||
| 		/* Read header. */ | ||||
|  | ||||
| 		const auto& bytes = decode(readRawBits(6*10)); | ||||
|  | ||||
| 		_sector->logicalSector = bytes[2]; | ||||
| 		_sector->logicalSide = 0; | ||||
| 		_sector->logicalTrack = bytes[0]; | ||||
|  | ||||
| 		uint16_t wantChecksum = bytes.reader().seek(4).read_be16(); | ||||
| 		uint16_t gotChecksum = crc16(CCITT_POLY, 0xef21, bytes.slice(0, 4)); | ||||
| 		if (wantChecksum == gotChecksum) | ||||
| 			_sector->status = Sector::DATA_MISSING; /* unintuitive but correct */ | ||||
| 	} | ||||
|  | ||||
|     void decodeDataRecord() | ||||
| 	{ | ||||
| 		/* Skip sync bits ID byte. */ | ||||
|  | ||||
| 		readRawBits(24); | ||||
|  | ||||
| 		const auto& bytes = decode(readRawBits((F85_SECTOR_LENGTH+3)*10)) | ||||
| 			.slice(0, F85_SECTOR_LENGTH+3); | ||||
| 		ByteReader br(bytes); | ||||
|  | ||||
| 		_sector->data = br.read(F85_SECTOR_LENGTH); | ||||
| 		uint16_t wantChecksum = br.read_be16(); | ||||
| 		uint16_t gotChecksum = crc16(CCITT_POLY, 0xbf84, _sector->data); | ||||
| 		_sector->status = (wantChecksum == gotChecksum) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createDurangoF85Decoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new DurangoF85Decoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										10
									
								
								arch/f85/f85.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										10
									
								
								arch/f85/f85.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,10 @@ | ||||
| #ifndef F85_H | ||||
| #define F85_H | ||||
|  | ||||
| #define F85_SECTOR_RECORD 0xffffce /* 1111 1111 1111 1111 1100 1110 */ | ||||
| #define F85_DATA_RECORD 0xffffcb /* 1111 1111 1111 1111 1100 1101 */ | ||||
| #define F85_SECTOR_LENGTH    512 | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createDurangoF85Decoder(const DecoderProto& config); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										4
									
								
								arch/f85/f85.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								arch/f85/f85.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| message F85DecoderProto {} | ||||
|  | ||||
| @@ -2,14 +2,12 @@ | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "protocol.h" | ||||
| #include "record.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "fb100.h" | ||||
| #include "crc.h" | ||||
| #include "bytes.h" | ||||
| #include "decoders/rawbits.h" | ||||
| #include "track.h" | ||||
| #include "fmt/format.h" | ||||
| #include <string.h> | ||||
| #include <algorithm> | ||||
| @@ -99,37 +97,52 @@ static uint16_t checksum(const Bytes& bytes) | ||||
|     return (crchi << 8) | crclo; | ||||
| } | ||||
| 
 | ||||
| AbstractDecoder::RecordType Fb100Decoder::advanceToNextRecord() | ||||
| class Fb100Decoder : public AbstractDecoder | ||||
| { | ||||
| 	const FluxMatcher* matcher = nullptr; | ||||
| 	_sector->clock = _fmr->seekToPattern(SECTOR_ID_PATTERN, matcher); | ||||
|     if (matcher == &SECTOR_ID_PATTERN) | ||||
| 		return RecordType::SECTOR_RECORD; | ||||
| 	return RecordType::UNKNOWN_RECORD; | ||||
| public: | ||||
| 	Fb100Decoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
| 
 | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(SECTOR_ID_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_ID_PATTERN) | ||||
| 			return RecordType::SECTOR_RECORD; | ||||
| 		return RecordType::UNKNOWN_RECORD; | ||||
| 	} | ||||
| 
 | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		auto rawbits = readRawBits(FB100_RECORD_SIZE*16); | ||||
| 
 | ||||
| 		const Bytes bytes = decodeFmMfm(rawbits).slice(0, FB100_RECORD_SIZE); | ||||
| 		ByteReader br(bytes); | ||||
| 		br.seek(1); | ||||
| 		const Bytes id = br.read(FB100_ID_SIZE); | ||||
| 		uint16_t wantIdCrc = br.read_be16(); | ||||
| 		uint16_t gotIdCrc = checksum(id); | ||||
| 		const Bytes payload = br.read(FB100_PAYLOAD_SIZE); | ||||
| 		uint16_t wantPayloadCrc = br.read_be16(); | ||||
| 		uint16_t gotPayloadCrc = checksum(payload); | ||||
| 
 | ||||
| 		if (wantIdCrc != gotIdCrc) | ||||
| 			return; | ||||
| 
 | ||||
| 		uint8_t abssector = id[2]; | ||||
| 		_sector->logicalTrack = abssector >> 1; | ||||
| 		_sector->logicalSide = 0; | ||||
| 		_sector->logicalSector = abssector & 1; | ||||
| 		_sector->data.writer().append(id.slice(5, 12)).append(payload); | ||||
| 
 | ||||
| 		_sector->status = (wantPayloadCrc == gotPayloadCrc) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
| }; | ||||
| 
 | ||||
| std::unique_ptr<AbstractDecoder> createFb100Decoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new Fb100Decoder(config)); | ||||
| } | ||||
| 
 | ||||
| void Fb100Decoder::decodeSectorRecord() | ||||
| { | ||||
|     auto rawbits = readRawBits(FB100_RECORD_SIZE*16); | ||||
| 
 | ||||
|     const Bytes bytes = decodeFmMfm(rawbits).slice(0, FB100_RECORD_SIZE); | ||||
|     ByteReader br(bytes); | ||||
|     br.seek(1); | ||||
|     const Bytes id = br.read(FB100_ID_SIZE); | ||||
|     uint16_t wantIdCrc = br.read_be16(); | ||||
|     uint16_t gotIdCrc = checksum(id); | ||||
|     const Bytes payload = br.read(FB100_PAYLOAD_SIZE); | ||||
|     uint16_t wantPayloadCrc = br.read_be16(); | ||||
|     uint16_t gotPayloadCrc = checksum(payload); | ||||
| 
 | ||||
|     if (wantIdCrc != gotIdCrc) | ||||
|         return; | ||||
| 
 | ||||
|     uint8_t abssector = id[2]; | ||||
|     _sector->logicalTrack = abssector >> 1; | ||||
|     _sector->logicalSide = 0; | ||||
|     _sector->logicalSector = abssector & 1; | ||||
|     _sector->data.writer().append(id.slice(5, 12)).append(payload); | ||||
| 
 | ||||
|     _sector->status = (wantPayloadCrc == gotPayloadCrc) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| } | ||||
							
								
								
									
										11
									
								
								arch/fb100/fb100.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								arch/fb100/fb100.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,11 @@ | ||||
| #ifndef FB100_H | ||||
| #define FB100_H | ||||
|  | ||||
| #define FB100_RECORD_SIZE 0x516 /* bytes */ | ||||
| #define FB100_ID_SIZE 17 | ||||
| #define FB100_PAYLOAD_SIZE 0x500 | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createFb100Decoder(const DecoderProto& config); | ||||
|  | ||||
| #endif | ||||
|  | ||||
							
								
								
									
										4
									
								
								arch/fb100/fb100.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								arch/fb100/fb100.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| message Fb100DecoderProto {} | ||||
|  | ||||
							
								
								
									
										228
									
								
								arch/ibm/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										228
									
								
								arch/ibm/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,228 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "ibm.h" | ||||
| #include "crc.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "sector.h" | ||||
| #include "arch/ibm/ibm.pb.h" | ||||
| #include "proto.h" | ||||
| #include <string.h> | ||||
|  | ||||
| static_assert(std::is_trivially_copyable<IbmIdam>::value, | ||||
| 		"IbmIdam is not trivially copyable"); | ||||
|  | ||||
| /* | ||||
|  * The markers at the beginning of records are special, and have | ||||
|  * missing clock pulses, allowing them to be found by the logic. | ||||
|  *  | ||||
|  * IAM record: | ||||
|  * flux:   XXXX-XXX-XXXX-X- = 0xf77a | ||||
|  * clock:  X X - X - X X X  = 0xd7 | ||||
|  * data:    X X X X X X - - = 0xfc | ||||
|  *  | ||||
|  * (We just ignore this one --- it's useless and optional.) | ||||
|  */ | ||||
|  | ||||
| /*  | ||||
|  * IDAM record: | ||||
|  * flux:   XXXX-X-X-XXXXXX- = 0xf57e | ||||
|  * clock:  X X - - - X X X  = 0xc7 | ||||
|  * data:    X X X X X X X - = 0xfe | ||||
|  */ | ||||
| const FluxPattern FM_IDAM_PATTERN(16, 0xf57e); | ||||
|  | ||||
| /*  | ||||
|  * DAM1 record: | ||||
|  * flux:   XXXX-X-X-XX-X-X- = 0xf56a | ||||
|  * clock:  X X - - - X X X  = 0xc7 | ||||
|  * data:    X X X X X - - - = 0xf8 | ||||
|  */ | ||||
| const FluxPattern FM_DAM1_PATTERN(16, 0xf56a); | ||||
|  | ||||
| /*  | ||||
|  * DAM2 record: | ||||
|  * flux:   XXXX-X-X-XX-XXXX = 0xf56f | ||||
|  * clock:  X X - - - X X X  = 0xc7 | ||||
|  * data:    X X X X X - X X = 0xfb | ||||
|  */ | ||||
| const FluxPattern FM_DAM2_PATTERN(16, 0xf56f); | ||||
|  | ||||
| /*  | ||||
|  * TRS80DAM1 record: | ||||
|  * flux:   XXXX-X-X-XX-X-XX = 0xf56b | ||||
|  * clock:  X X - - - X X X  = 0xc7 | ||||
|  * data:    X X X X X - - X = 0xf9 | ||||
|  */ | ||||
| const FluxPattern FM_TRS80DAM1_PATTERN(16, 0xf56b); | ||||
|  | ||||
| /*  | ||||
|  * TRS80DAM2 record: | ||||
|  * flux:   XXXX-X-X-XX-XXX- = 0xf56e | ||||
|  * clock:  X X - - - X X X  = 0xc7 | ||||
|  * data:    X X X X X - X - = 0xfa | ||||
|  */ | ||||
| const FluxPattern FM_TRS80DAM2_PATTERN(16, 0xf56e); | ||||
|  | ||||
| /* MFM record separator: | ||||
|  * 0xA1 is: | ||||
|  * data:    1  0  1  0  0  0  0  1  = 0xa1 | ||||
|  * mfm:     01 00 01 00 10 10 10 01 = 0x44a9 | ||||
|  * special: 01 00 01 00 10 00 10 01 = 0x4489 | ||||
|  *                       ^^^^^ | ||||
|  * When shifted out of phase, the special 0xa1 byte becomes an illegal | ||||
|  * encoding (you can't do 10 00). So this can't be spoofed by user data. | ||||
|  *  | ||||
|  * shifted: 10 00 10 01 00 01 00 1 | ||||
|  *  | ||||
|  * It's repeated three times. | ||||
|  */ | ||||
| const FluxPattern MFM_PATTERN(48, 0x448944894489LL); | ||||
|  | ||||
| const FluxMatchers ANY_RECORD_PATTERN( | ||||
|     { | ||||
|         &MFM_PATTERN, | ||||
|         &FM_IDAM_PATTERN, | ||||
|         &FM_DAM1_PATTERN, | ||||
|         &FM_DAM2_PATTERN, | ||||
|         &FM_TRS80DAM1_PATTERN, | ||||
|         &FM_TRS80DAM2_PATTERN, | ||||
|     } | ||||
| ); | ||||
|  | ||||
| class IbmDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
|     IbmDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config), | ||||
| 		_config(config.ibm()) | ||||
|     {} | ||||
|  | ||||
|     RecordType advanceToNextRecord() override | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
|  | ||||
| 		/* If this is the MFM prefix byte, the the decoder is going to expect three | ||||
| 		 * extra bytes on the front of the header. */ | ||||
| 		_currentHeaderLength = (matcher == &MFM_PATTERN) ? 3 : 0; | ||||
|  | ||||
| 		Fluxmap::Position here = tell(); | ||||
| 		resetFluxDecoder(); | ||||
| 		if (_currentHeaderLength > 0) | ||||
| 			readRawBits(_currentHeaderLength*16); | ||||
| 		auto idbits = readRawBits(16); | ||||
| 		const Bytes idbytes = decodeFmMfm(idbits); | ||||
| 		uint8_t id = idbytes.slice(0, 1)[0]; | ||||
| 		if (eof()) | ||||
| 			return RecordType::UNKNOWN_RECORD; | ||||
| 		seek(here); | ||||
| 		 | ||||
| 		switch (id) | ||||
| 		{ | ||||
| 			case IBM_IDAM: | ||||
| 				return RecordType::SECTOR_RECORD; | ||||
|  | ||||
| 			case IBM_DAM1: | ||||
| 			case IBM_DAM2: | ||||
| 			case IBM_TRS80DAM1: | ||||
| 			case IBM_TRS80DAM2: | ||||
| 				return RecordType::DATA_RECORD; | ||||
| 		} | ||||
| 		return RecordType::UNKNOWN_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() override | ||||
| 	{ | ||||
| 		unsigned recordSize = _currentHeaderLength + IBM_IDAM_LEN; | ||||
| 		auto bits = readRawBits(recordSize*16); | ||||
| 		auto bytes = decodeFmMfm(bits).slice(0, recordSize); | ||||
|  | ||||
| 		IbmDecoderProto::TrackdataProto trackdata; | ||||
| 		getTrackFormat(trackdata, _sector->physicalCylinder, _sector->physicalHead); | ||||
|  | ||||
| 		ByteReader br(bytes); | ||||
| 		br.seek(_currentHeaderLength); | ||||
| 		br.read_8(); /* skip ID byte */ | ||||
| 		_sector->logicalTrack = br.read_8(); | ||||
| 		_sector->logicalSide = br.read_8(); | ||||
| 		_sector->logicalSector = br.read_8(); | ||||
| 		_currentSectorSize = 1 << (br.read_8() + 7); | ||||
| 		uint16_t wantCrc = br.read_be16(); | ||||
| 		uint16_t gotCrc = crc16(CCITT_POLY, bytes.slice(0, _currentHeaderLength + 5)); | ||||
| 		if (wantCrc == gotCrc) | ||||
| 			_sector->status = Sector::DATA_MISSING; /* correct but unintuitive */ | ||||
|  | ||||
| 		if (trackdata.swap_sides()) | ||||
| 			_sector->logicalSide ^= 1; | ||||
| 		if (trackdata.ignore_side_byte()) | ||||
| 			_sector->logicalSide = _sector->physicalHead; | ||||
| 		if (trackdata.ignore_track_byte()) | ||||
| 			_sector->logicalTrack = _sector->physicalCylinder; | ||||
| 	} | ||||
|  | ||||
|     void decodeDataRecord() override | ||||
| 	{ | ||||
| 		unsigned recordLength = _currentHeaderLength + _currentSectorSize + 3; | ||||
| 		auto bits = readRawBits(recordLength*16); | ||||
| 		auto bytes = decodeFmMfm(bits).slice(0, recordLength); | ||||
|  | ||||
| 		ByteReader br(bytes); | ||||
| 		br.seek(_currentHeaderLength); | ||||
| 		br.read_8(); /* skip ID byte */ | ||||
|  | ||||
| 		_sector->data = br.read(_currentSectorSize); | ||||
| 		uint16_t wantCrc = br.read_be16(); | ||||
| 		uint16_t gotCrc = crc16(CCITT_POLY, bytes.slice(0, recordLength-2)); | ||||
| 		_sector->status = (wantCrc == gotCrc) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
|  | ||||
| 	std::set<unsigned> requiredSectors(unsigned cylinder, unsigned head) const override | ||||
| 	{ | ||||
| 		IbmDecoderProto::TrackdataProto trackdata; | ||||
| 		getTrackFormat(trackdata, cylinder, head); | ||||
|  | ||||
| 		std::set<unsigned> s; | ||||
| 		if (trackdata.has_sectors()) | ||||
| 		{ | ||||
| 			for (int sectorId : trackdata.sectors().sector()) | ||||
| 				s.insert(sectorId); | ||||
| 		} | ||||
| 		else if (trackdata.has_sector_range()) | ||||
| 		{ | ||||
| 			int sectorId = trackdata.sector_range().min_sector(); | ||||
| 			while (sectorId <= trackdata.sector_range().max_sector()) | ||||
| 			{ | ||||
| 				s.insert(sectorId); | ||||
| 				sectorId++; | ||||
| 			} | ||||
| 		} | ||||
| 		return s; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	void getTrackFormat(IbmDecoderProto::TrackdataProto& trackdata, unsigned cylinder, unsigned head) const | ||||
| 	{ | ||||
| 		trackdata.Clear(); | ||||
| 		for (const auto& f : _config.trackdata()) | ||||
| 		{ | ||||
| 			if (f.has_cylinder() && (f.cylinder() != cylinder)) | ||||
| 				continue; | ||||
| 			if (f.has_head() && (f.head() != head)) | ||||
| 				continue; | ||||
|  | ||||
| 			trackdata.MergeFrom(f); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const IbmDecoderProto& _config; | ||||
|     unsigned _currentSectorSize; | ||||
|     unsigned _currentHeaderLength; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createIbmDecoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new IbmDecoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										313
									
								
								arch/ibm/encoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										313
									
								
								arch/ibm/encoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,313 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "ibm.h" | ||||
| #include "crc.h" | ||||
| #include "writer.h" | ||||
| #include "image.h" | ||||
| #include "arch/ibm/ibm.pb.h" | ||||
| #include "lib/encoders/encoders.pb.h" | ||||
| #include "fmt/format.h" | ||||
| #include <ctype.h> | ||||
|  | ||||
| /* IAM record separator: | ||||
|  * 0xC2 is: | ||||
|  * data:    1  1  0  0  0  0  1  0  = 0xc2 | ||||
|  * mfm:     01 01 00 10 10 10 01 00 = 0x5254 | ||||
|  * special: 01 01 00 10 00 10 01 00 = 0x5224 | ||||
|  */ | ||||
| #define MFM_IAM_SEPARATOR 0x5224 | ||||
|  | ||||
| /* FM IAM record: | ||||
|  * flux:   XXXX-XXX-XXXX-X- = 0xf77a | ||||
|  * clock:  X X - X - X X X  = 0xd7 | ||||
|  * data:    X X X X X X - - = 0xfc | ||||
|  */ | ||||
| #define FM_IAM_RECORD 0xf77a | ||||
|  | ||||
| /* MFM IAM record: | ||||
|  * data:   1  1  1  1  1  1  0  0  = 0xfc | ||||
|  * flux:   01 01 01 01 01 01 00 10 = 0x5552 | ||||
|  */ | ||||
| #define MFM_IAM_RECORD 0x5552 | ||||
|  | ||||
| /* MFM record separator: | ||||
|  * 0xA1 is: | ||||
|  * data:    1  0  1  0  0  0  0  1  = 0xa1 | ||||
|  * mfm:     01 00 01 00 10 10 10 01 = 0x44a9 | ||||
|  * special: 01 00 01 00 10 00 10 01 = 0x4489 | ||||
|  *                       ^^^^^ | ||||
|  * When shifted out of phase, the special 0xa1 byte becomes an illegal | ||||
|  * encoding (you can't do 10 00). So this can't be spoofed by user data. | ||||
|  *  | ||||
|  * shifted: 10 00 10 01 00 01 00 1 | ||||
|  *  | ||||
|  * It's repeated three times. | ||||
|  */ | ||||
| #define MFM_RECORD_SEPARATOR 0x4489 | ||||
| #define MFM_RECORD_SEPARATOR_BYTE 0xa1 | ||||
|  | ||||
| /* MFM IDAM byte: | ||||
|  * data:    1  1  1  1  1  1  1  0  = 0xfe | ||||
|  * mfm:     01 01 01 01 01 01 01 00 = 0x5554 | ||||
|  */ | ||||
|  | ||||
| /* MFM DAM byte: | ||||
|  * data:    1  1  1  1  1  0  1  1  = 0xfb | ||||
|  * mfm:     01 01 01 01 01 00 01 01 = 0x5545 | ||||
|  */ | ||||
|  | ||||
| static uint8_t decodeUint16(uint16_t raw) | ||||
| { | ||||
| 	Bytes b; | ||||
| 	ByteWriter bw(b); | ||||
| 	bw.write_be16(raw); | ||||
| 	return decodeFmMfm(b.toBits())[0]; | ||||
| } | ||||
|  | ||||
| class IbmEncoder : public AbstractEncoder | ||||
| { | ||||
| public: | ||||
| 	IbmEncoder(const EncoderProto& config): | ||||
| 		AbstractEncoder(config), | ||||
| 		_config(config.ibm()) | ||||
| 	{} | ||||
|  | ||||
| private: | ||||
| 	void writeRawBits(uint32_t data, int width) | ||||
| 	{ | ||||
| 		_cursor += width; | ||||
| 		_lastBit = data & 1; | ||||
| 		for (int i=0; i<width; i++) | ||||
| 		{ | ||||
| 			unsigned pos = _cursor - i - 1; | ||||
| 			if (pos < _bits.size()) | ||||
| 				_bits[pos] = data & 1; | ||||
| 			data >>= 1; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	void getTrackFormat(IbmEncoderProto::TrackdataProto& trackdata, unsigned cylinder, unsigned head) | ||||
| 	{ | ||||
| 		trackdata.Clear(); | ||||
| 		for (const auto& f : _config.trackdata()) | ||||
| 		{ | ||||
| 			if (f.has_cylinder() && (f.cylinder() != cylinder)) | ||||
| 				continue; | ||||
| 			if (f.has_head() && (f.head() != head)) | ||||
| 				continue; | ||||
|  | ||||
| 			trackdata.MergeFrom(f); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	static std::set<unsigned> getSectorIds(const IbmEncoderProto::TrackdataProto& trackdata) | ||||
| 	{ | ||||
| 		std::set<unsigned> s; | ||||
| 		if (trackdata.has_sectors()) | ||||
| 		{ | ||||
| 			for (int sectorId : trackdata.sectors().sector()) | ||||
| 				s.insert(sectorId); | ||||
| 		} | ||||
| 		else if (trackdata.has_sector_range()) | ||||
| 		{ | ||||
| 			int sectorId = trackdata.sector_range().min_sector(); | ||||
| 			while (sectorId <= trackdata.sector_range().max_sector()) | ||||
| 			{ | ||||
| 				s.insert(sectorId); | ||||
| 				sectorId++; | ||||
| 			} | ||||
| 		} | ||||
| 		return s; | ||||
| 	} | ||||
|  | ||||
| public: | ||||
| 	std::vector<std::shared_ptr<Sector>> collectSectors(int physicalTrack, int physicalSide, const Image& image) override | ||||
| 	{ | ||||
| 		std::vector<std::shared_ptr<Sector>> sectors; | ||||
| 		IbmEncoderProto::TrackdataProto trackdata; | ||||
| 		getTrackFormat(trackdata, physicalTrack, physicalSide); | ||||
|  | ||||
| 		int logicalSide = physicalSide ^ trackdata.swap_sides(); | ||||
| 		for (int sectorId : getSectorIds(trackdata)) | ||||
|         { | ||||
| 			const auto& sector = image.get(physicalTrack, logicalSide, sectorId); | ||||
| 			if (sector) | ||||
| 				sectors.push_back(sector); | ||||
|         } | ||||
|  | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
|     std::unique_ptr<Fluxmap> encode(int physicalTrack, int physicalSide, | ||||
| 			const std::vector<std::shared_ptr<Sector>>& sectors, const Image& image) override | ||||
| 	{ | ||||
| 		IbmEncoderProto::TrackdataProto trackdata; | ||||
| 		getTrackFormat(trackdata, physicalTrack, physicalSide); | ||||
|  | ||||
| 		auto writeBytes = [&](const Bytes& bytes) | ||||
| 		{ | ||||
| 			if (trackdata.use_fm()) | ||||
| 				encodeFm(_bits, _cursor, bytes); | ||||
| 			else | ||||
| 				encodeMfm(_bits, _cursor, bytes, _lastBit); | ||||
| 		}; | ||||
|  | ||||
| 		auto writeFillerRawBytes = [&](int count, uint16_t byte) | ||||
| 		{ | ||||
| 			for (int i=0; i<count; i++) | ||||
| 				writeRawBits(byte, 16); | ||||
| 		}; | ||||
|  | ||||
| 		auto writeFillerBytes = [&](int count, uint8_t byte) | ||||
| 		{ | ||||
| 			Bytes b { byte }; | ||||
| 			for (int i=0; i<count; i++) | ||||
| 				writeBytes(b); | ||||
| 		}; | ||||
|  | ||||
| 		double clockRateUs = 1e3 / trackdata.clock_rate_khz(); | ||||
| 		if (!trackdata.use_fm()) | ||||
| 			clockRateUs /= 2.0; | ||||
| 		int bitsPerRevolution = (trackdata.track_length_ms() * 1000.0) / clockRateUs; | ||||
| 		_bits.resize(bitsPerRevolution); | ||||
| 		_cursor = 0; | ||||
|  | ||||
| 		uint8_t idamUnencoded = decodeUint16(trackdata.idam_byte()); | ||||
| 		uint8_t damUnencoded = decodeUint16(trackdata.dam_byte()); | ||||
|  | ||||
| 		uint8_t sectorSize = 0; | ||||
| 		{ | ||||
| 			int s = trackdata.sector_size() >> 7; | ||||
| 			while (s > 1) | ||||
| 			{ | ||||
| 				s >>= 1; | ||||
| 				sectorSize += 1; | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		uint16_t gapFill = trackdata.gap_fill_byte(); | ||||
|  | ||||
| 		writeFillerRawBytes(trackdata.gap0(), gapFill); | ||||
| 		if (trackdata.emit_iam()) | ||||
| 		{ | ||||
| 			writeFillerBytes(trackdata.use_fm() ? 6 : 12, 0x00); | ||||
| 			if (!trackdata.use_fm()) | ||||
| 			{ | ||||
| 				for (int i=0; i<3; i++) | ||||
| 					writeRawBits(MFM_IAM_SEPARATOR, 16); | ||||
| 			} | ||||
| 			writeRawBits(trackdata.use_fm() ? FM_IAM_RECORD : MFM_IAM_RECORD, 16); | ||||
| 			writeFillerRawBytes(trackdata.gap1(), gapFill); | ||||
| 		} | ||||
|  | ||||
| 		int logicalSide = physicalSide ^ trackdata.swap_sides(); | ||||
| 		bool first = true; | ||||
| 		for (int sectorId : trackdata.sectors().sector()) | ||||
| 		{ | ||||
| 			if (!first) | ||||
| 				writeFillerRawBytes(trackdata.gap3(), gapFill); | ||||
| 			first = false; | ||||
|  | ||||
| 			const auto& sectorData = image.get(physicalTrack, logicalSide, sectorId); | ||||
| 			if (!sectorData) | ||||
| 				continue; | ||||
|  | ||||
| 			/* Writing the sector and data records are fantastically annoying. | ||||
| 			 * The CRC is calculated from the *very start* of the record, and | ||||
| 			 * include the malformed marker bytes. Our encoder doesn't know | ||||
| 			 * about this, of course, with the result that we have to construct | ||||
| 			 * the unencoded header, calculate the checksum, and then use the | ||||
| 			 * same logic to emit the bytes which require special encoding | ||||
| 			 * before encoding the rest of the header normally. */ | ||||
|  | ||||
| 			{ | ||||
| 				Bytes header; | ||||
| 				ByteWriter bw(header); | ||||
|  | ||||
| 				writeFillerBytes(trackdata.use_fm() ? 6 : 12, 0x00); | ||||
| 				if (!trackdata.use_fm()) | ||||
| 				{ | ||||
| 					for (int i=0; i<3; i++) | ||||
| 						bw.write_8(MFM_RECORD_SEPARATOR_BYTE); | ||||
| 				} | ||||
| 				bw.write_8(idamUnencoded); | ||||
| 				bw.write_8(sectorData->logicalTrack); | ||||
| 				bw.write_8(sectorData->logicalSide); | ||||
| 				bw.write_8(sectorData->logicalSector); | ||||
| 				bw.write_8(sectorSize); | ||||
| 				uint16_t crc = crc16(CCITT_POLY, header); | ||||
| 				bw.write_be16(crc); | ||||
|  | ||||
| 				int conventionalHeaderStart = 0; | ||||
| 				if (!trackdata.use_fm()) | ||||
| 				{ | ||||
| 					for (int i=0; i<3; i++) | ||||
| 						writeRawBits(MFM_RECORD_SEPARATOR, 16); | ||||
| 					conventionalHeaderStart += 3; | ||||
|  | ||||
| 				} | ||||
| 				writeRawBits(trackdata.idam_byte(), 16); | ||||
| 				conventionalHeaderStart += 1; | ||||
|  | ||||
| 				writeBytes(header.slice(conventionalHeaderStart)); | ||||
| 			} | ||||
|  | ||||
| 			writeFillerRawBytes(trackdata.gap2(), gapFill); | ||||
|  | ||||
| 			{ | ||||
| 				Bytes data; | ||||
| 				ByteWriter bw(data); | ||||
|  | ||||
| 				writeFillerBytes(trackdata.use_fm() ? 6 : 12, 0x00); | ||||
| 				if (!trackdata.use_fm()) | ||||
| 				{ | ||||
| 					for (int i=0; i<3; i++) | ||||
| 						bw.write_8(MFM_RECORD_SEPARATOR_BYTE); | ||||
| 				} | ||||
| 				bw.write_8(damUnencoded); | ||||
|  | ||||
| 				Bytes truncatedData = sectorData->data.slice(0, trackdata.sector_size()); | ||||
| 				bw += truncatedData; | ||||
| 				uint16_t crc = crc16(CCITT_POLY, data); | ||||
| 				bw.write_be16(crc); | ||||
|  | ||||
| 				int conventionalHeaderStart = 0; | ||||
| 				if (!trackdata.use_fm()) | ||||
| 				{ | ||||
| 					for (int i=0; i<3; i++) | ||||
| 						writeRawBits(MFM_RECORD_SEPARATOR, 16); | ||||
| 					conventionalHeaderStart += 3; | ||||
|  | ||||
| 				} | ||||
| 				writeRawBits(trackdata.dam_byte(), 16); | ||||
| 				conventionalHeaderStart += 1; | ||||
|  | ||||
| 				writeBytes(data.slice(conventionalHeaderStart)); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (_cursor >= _bits.size()) | ||||
| 			Error() << "track data overrun"; | ||||
| 		while (_cursor < _bits.size()) | ||||
| 			writeFillerRawBytes(1, gapFill); | ||||
|  | ||||
| 		std::unique_ptr<Fluxmap> fluxmap(new Fluxmap); | ||||
| 		fluxmap->appendBits(_bits, clockRateUs*1e3); | ||||
| 		return fluxmap; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const IbmEncoderProto& _config; | ||||
| 	std::vector<bool> _bits; | ||||
| 	unsigned _cursor; | ||||
| 	bool _lastBit; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractEncoder> createIbmEncoder(const EncoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractEncoder>(new IbmEncoder(config)); | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										37
									
								
								arch/ibm/ibm.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								arch/ibm/ibm.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,37 @@ | ||||
| #ifndef IBM_H | ||||
| #define IBM_H | ||||
|  | ||||
| /* IBM format (i.e. ordinary PC floppies). */ | ||||
|  | ||||
| #define IBM_MFM_SYNC   0xA1   /* sync byte for MFM */ | ||||
| #define IBM_IAM        0xFC   /* start-of-track record */ | ||||
| #define IBM_IAM_LEN    1      /* plus prologue */ | ||||
| #define IBM_IDAM       0xFE   /* sector header */ | ||||
| #define IBM_IDAM_LEN   7      /* plus prologue */ | ||||
| #define IBM_DAM1       0xF8   /* sector data (type 1) */ | ||||
| #define IBM_DAM2       0xFB   /* sector data (type 2) */ | ||||
| #define IBM_TRS80DAM1  0xF9   /* sector data (TRS-80 directory) */ | ||||
| #define IBM_TRS80DAM2  0xFA   /* sector data (TRS-80 directory) */ | ||||
| #define IBM_DAM_LEN    1      /* plus prologue and user data */ | ||||
|  | ||||
| /* Length of a DAM record is determined by the previous sector header. */ | ||||
|  | ||||
| struct IbmIdam | ||||
| { | ||||
|     uint8_t id; | ||||
|     uint8_t cylinder; | ||||
|     uint8_t side; | ||||
|     uint8_t sector; | ||||
|     uint8_t sectorSize; | ||||
|     uint8_t crc[2]; | ||||
| }; | ||||
|  | ||||
| class AbstractEncoder; | ||||
| class AbstractDecoder; | ||||
| class DecoderProto; | ||||
| class EncoderProto; | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createIbmDecoder(const DecoderProto& config); | ||||
| extern std::unique_ptr<AbstractEncoder> createIbmEncoder(const EncoderProto& config); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										68
									
								
								arch/ibm/ibm.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										68
									
								
								arch/ibm/ibm.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,68 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| import "lib/common.proto"; | ||||
|  | ||||
| message IbmDecoderProto { | ||||
| 	// Next: 10 | ||||
| 	message TrackdataProto { | ||||
| 		message SectorsProto { | ||||
| 			repeated int32 sector = 1            [(help) = "require these sectors to exist for a good read"]; | ||||
| 		} | ||||
| 		message SectorRangeProto { | ||||
| 			optional int32 min_sector = 1        [(help) = "require these sectors to exist for a good read"]; | ||||
| 			optional int32 max_sector = 2        [(help) = "require these sectors to exist for a good read"]; | ||||
| 		} | ||||
|  | ||||
| 		optional int32 cylinder = 7              [(help) = "if set, the format applies only to this track"]; | ||||
| 		optional int32 head = 8                  [(help) = "if set, the format applies only to this head"]; | ||||
|  | ||||
| 		optional bool ignore_side_byte = 2       [default = false, (help) = "ignore side byte in sector header"]; | ||||
| 		optional bool ignore_track_byte = 6      [default = false, (help) = "ignore track byte in sector header"]; | ||||
| 		optional bool swap_sides = 4             [default = false, (help) = "put logical side 1 on physical side 0"]; | ||||
|  | ||||
| 		oneof required_sectors { | ||||
| 			SectorsProto sectors = 5             [(help) = "require these sectors to exist for a good read"]; | ||||
| 			SectorRangeProto sector_range = 9    [(help) = "require these sectors to exist for a good read"]; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	repeated TrackdataProto trackdata = 1; | ||||
| } | ||||
|  | ||||
| message IbmEncoderProto { | ||||
| 	// Next: 20 | ||||
| 	message TrackdataProto { | ||||
| 		message SectorsProto { | ||||
| 			repeated int32 sector = 1		[(help) = "write these sectors (in order) on each track"]; | ||||
| 		} | ||||
| 		message SectorRangeProto { | ||||
| 			optional int32 min_sector = 1   [(help) = "write these sectors (in order) on each track"]; | ||||
| 			optional int32 max_sector = 2   [(help) = "write these sectors (in order) on each track"]; | ||||
| 		} | ||||
|  | ||||
| 		optional int32 cylinder = 15        [(help) = "if set, the format applies only to this track"]; | ||||
| 		optional int32 head = 16            [(help) = "if set, the format applies only to this head"]; | ||||
|  | ||||
| 		optional double track_length_ms = 1 [(help) = "length of track"]; | ||||
| 		optional int32 sector_size = 2      [default=512, (help) = "number of bytes per sector"]; | ||||
| 		optional bool emit_iam = 3          [default=true, (help) = "whether to emit an IAM record"]; | ||||
| 		optional double clock_rate_khz = 5  [(help) = "data clock rate"]; | ||||
| 		optional bool use_fm = 6            [default=false, (help) = "whether to use FM encoding rather than MFM"]; | ||||
| 		optional int32 idam_byte = 7        [default=0x5554, (help) = "16-bit raw bit pattern of IDAM byte"]; | ||||
| 		optional int32 dam_byte = 8         [default=0x5545, (help) = "16-bit raw bit pattern of DAM byte"]; | ||||
| 		optional int32 gap0 = 9             [default=80, (help) = "size of gap 1 (the post-index gap)"]; | ||||
| 		optional int32 gap1 = 10            [default=50, (help) = "size of gap 2 (the post-ID gap)"]; | ||||
| 		optional int32 gap2 = 11            [default=22, (help) = "size of gap 3 (the pre-data gap)"]; | ||||
| 		optional int32 gap3 = 12            [default=80, (help) = "size of gap 4 (the post-data or format gap)"]; | ||||
| 		optional bool swap_sides = 14       [default=false, (help) = "swap side bytes when writing"]; | ||||
| 		optional int32 gap_fill_byte = 18   [default=0x9254, (help) = "16-bit raw bit pattern of gap fill byte"]; | ||||
|  | ||||
| 		oneof required_sectors { | ||||
| 			SectorsProto sectors = 17            [(help) = "require these sectors to exist for a good read"]; | ||||
| 			SectorRangeProto sector_range = 19   [(help) = "require these sectors to exist for a good read"]; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	repeated TrackdataProto trackdata = 1; | ||||
| } | ||||
|  | ||||
| @@ -2,10 +2,8 @@ | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "protocol.h" | ||||
| #include "record.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "track.h" | ||||
| #include "macintosh.h" | ||||
| #include "bytes.h" | ||||
| #include "fmt/format.h" | ||||
| @@ -26,7 +24,7 @@ static int decode_data_gcr(uint8_t gcr) | ||||
| 		#undef GCR_ENTRY | ||||
|     } | ||||
|     return -1; | ||||
| }; | ||||
| } | ||||
| 
 | ||||
| /* This is extremely inspired by the MESS implementation, written by Nathan Woods
 | ||||
|  * and R. Belmont: https://github.com/mamedev/mame/blob/4263a71e64377db11392c458b580c5ae83556bc7/src/lib/formats/ap_dsk35.cpp
 | ||||
| @@ -118,64 +116,103 @@ static Bytes decode_crazy_data(const Bytes& input, Sector::Status& status) | ||||
| uint8_t decode_side(uint8_t side) | ||||
| { | ||||
|     /* Mac disks, being weird, use the side byte to encode both the side (in
 | ||||
|      * bit 5) and also whether we're above track 0x3f (in bit 6). | ||||
|      * bit 5) and also whether we're above track 0x3f (in bit 0). | ||||
|      */ | ||||
| 
 | ||||
|     return !!(side & 0x40); | ||||
|     return !!(side & 0x20); | ||||
| } | ||||
| 
 | ||||
| AbstractDecoder::RecordType MacintoshDecoder::advanceToNextRecord() | ||||
| class MacintoshDecoder : public AbstractDecoder | ||||
| { | ||||
| 	const FluxMatcher* matcher = nullptr; | ||||
| 	_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
| 	if (matcher == &SECTOR_RECORD_PATTERN) | ||||
| 		return SECTOR_RECORD; | ||||
| 	if (matcher == &DATA_RECORD_PATTERN) | ||||
| 		return DATA_RECORD; | ||||
| 	return UNKNOWN_RECORD; | ||||
| } | ||||
| public: | ||||
| 	MacintoshDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
| 
 | ||||
| void MacintoshDecoder::decodeSectorRecord() | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_RECORD_PATTERN) | ||||
| 			return SECTOR_RECORD; | ||||
| 		if (matcher == &DATA_RECORD_PATTERN) | ||||
| 			return DATA_RECORD; | ||||
| 		return UNKNOWN_RECORD; | ||||
| 	} | ||||
| 
 | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		/* Skip ID (as we know it's a MAC_SECTOR_RECORD). */ | ||||
| 		readRawBits(24); | ||||
| 
 | ||||
| 		/* Read header. */ | ||||
| 
 | ||||
| 		auto header = toBytes(readRawBits(7*8)).slice(0, 7); | ||||
| 					 | ||||
| 		uint8_t encodedTrack = decode_data_gcr(header[0]); | ||||
| 		if (encodedTrack != (_sector->physicalCylinder & 0x3f)) | ||||
| 			return; | ||||
| 					 | ||||
| 		uint8_t encodedSector = decode_data_gcr(header[1]); | ||||
| 		uint8_t encodedSide = decode_data_gcr(header[2]); | ||||
| 		uint8_t formatByte = decode_data_gcr(header[3]); | ||||
| 		uint8_t wantedsum = decode_data_gcr(header[4]); | ||||
| 
 | ||||
| 		if (encodedSector > 11) | ||||
| 			return; | ||||
| 
 | ||||
| 		_sector->logicalTrack = _sector->physicalCylinder; | ||||
| 		_sector->logicalSide = decode_side(encodedSide); | ||||
| 		_sector->logicalSector = encodedSector; | ||||
| 		uint8_t gotsum = (encodedTrack ^ encodedSector ^ encodedSide ^ formatByte) & 0x3f; | ||||
| 		if (wantedsum == gotsum) | ||||
| 			_sector->status = Sector::DATA_MISSING; /* unintuitive but correct */ | ||||
| 	} | ||||
| 
 | ||||
|     void decodeDataRecord() | ||||
| 	{ | ||||
| 		auto id = toBytes(readRawBits(24)).reader().read_be24(); | ||||
| 		if (id != MAC_DATA_RECORD) | ||||
| 			return; | ||||
| 
 | ||||
| 		/* Read data. */ | ||||
| 
 | ||||
| 		readRawBits(8); /* skip spare byte */ | ||||
| 		auto inputbuffer = toBytes(readRawBits(MAC_ENCODED_SECTOR_LENGTH*8)) | ||||
| 			.slice(0, MAC_ENCODED_SECTOR_LENGTH); | ||||
| 
 | ||||
| 		for (unsigned i=0; i<inputbuffer.size(); i++) | ||||
| 			inputbuffer[i] = decode_data_gcr(inputbuffer[i]); | ||||
| 			 | ||||
| 		_sector->status = Sector::BAD_CHECKSUM; | ||||
| 		Bytes userData = decode_crazy_data(inputbuffer, _sector->status); | ||||
| 		_sector->data.clear(); | ||||
| 		_sector->data.writer().append(userData.slice(12, 512)).append(userData.slice(0, 12)); | ||||
| 	} | ||||
| 
 | ||||
| 	std::set<unsigned> requiredSectors(unsigned cylinder, unsigned head) const | ||||
| 	{ | ||||
| 		int count; | ||||
| 		if (cylinder < 16) | ||||
| 			count = 12; | ||||
| 		else if (cylinder < 32) | ||||
| 			count = 11; | ||||
| 		else if (cylinder < 48) | ||||
| 			count = 10; | ||||
| 		else if (cylinder < 64) | ||||
| 			count = 9; | ||||
| 		else | ||||
| 			count = 8; | ||||
| 
 | ||||
| 		std::set<unsigned> sectors; | ||||
| 		while (count--) | ||||
| 			sectors.insert(count); | ||||
| 		return sectors; | ||||
| 	} | ||||
| }; | ||||
| 
 | ||||
| std::unique_ptr<AbstractDecoder> createMacintoshDecoder(const DecoderProto& config) | ||||
| { | ||||
|     /* Skip ID (as we know it's a MAC_SECTOR_RECORD). */ | ||||
|     readRawBits(24); | ||||
| 
 | ||||
|     /* Read header. */ | ||||
| 
 | ||||
|     auto header = toBytes(readRawBits(7*8)).slice(0, 7); | ||||
|                  | ||||
|     uint8_t encodedTrack = decode_data_gcr(header[0]); | ||||
|     if (encodedTrack != (_track->physicalTrack & 0x3f)) | ||||
|         return; | ||||
|                  | ||||
|     uint8_t encodedSector = decode_data_gcr(header[1]); | ||||
|     uint8_t encodedSide = decode_data_gcr(header[2]); | ||||
|     uint8_t formatByte = decode_data_gcr(header[3]); | ||||
|     uint8_t wantedsum = decode_data_gcr(header[4]); | ||||
| 
 | ||||
|     _sector->logicalTrack = _track->physicalTrack; | ||||
|     _sector->logicalSide = decode_side(encodedSide); | ||||
|     _sector->logicalSector = encodedSector; | ||||
|     uint8_t gotsum = (encodedTrack ^ encodedSector ^ encodedSide ^ formatByte) & 0x3f; | ||||
|     if (wantedsum == gotsum) | ||||
|         _sector->status = Sector::DATA_MISSING; /* unintuitive but correct */ | ||||
| 	return std::unique_ptr<AbstractDecoder>(new MacintoshDecoder(config)); | ||||
| } | ||||
| 
 | ||||
| void MacintoshDecoder::decodeDataRecord() | ||||
| { | ||||
|     auto id = toBytes(readRawBits(24)).reader().read_be24(); | ||||
|     if (id != MAC_DATA_RECORD) | ||||
|         return; | ||||
| 
 | ||||
|     /* Read data. */ | ||||
| 
 | ||||
|     readRawBits(8); /* skip spare byte */ | ||||
|     auto inputbuffer = toBytes(readRawBits(MAC_ENCODED_SECTOR_LENGTH*8)) | ||||
|         .slice(0, MAC_ENCODED_SECTOR_LENGTH); | ||||
| 
 | ||||
|     for (unsigned i=0; i<inputbuffer.size(); i++) | ||||
|         inputbuffer[i] = decode_data_gcr(inputbuffer[i]); | ||||
|          | ||||
|     _sector->status = Sector::BAD_CHECKSUM; | ||||
|     _sector->data = decode_crazy_data(inputbuffer, _sector->status); | ||||
| } | ||||
							
								
								
									
										263
									
								
								arch/macintosh/encoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										263
									
								
								arch/macintosh/encoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,263 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "macintosh.h" | ||||
| #include "crc.h" | ||||
| #include "writer.h" | ||||
| #include "image.h" | ||||
| #include "fmt/format.h" | ||||
| #include "lib/encoders/encoders.pb.h" | ||||
| #include "arch/macintosh/macintosh.pb.h" | ||||
| #include <ctype.h> | ||||
|  | ||||
| static bool lastBit; | ||||
|  | ||||
| static double clockRateUsForTrack(unsigned track) | ||||
| { | ||||
| 	if (track < 16) | ||||
| 		return 2.623; | ||||
| 	if (track < 32) | ||||
| 		return 2.861; | ||||
| 	if (track < 48) | ||||
| 		return 3.148; | ||||
| 	if (track < 64) | ||||
| 		return 3.497; | ||||
| 	return 3.934; | ||||
| } | ||||
|  | ||||
| static unsigned sectorsForTrack(unsigned track) | ||||
| { | ||||
| 	if (track < 16) | ||||
| 		return 12; | ||||
| 	if (track < 32) | ||||
| 		return 11; | ||||
| 	if (track < 48) | ||||
| 		return 10; | ||||
| 	if (track < 64) | ||||
| 		return 9; | ||||
| 	return 8; | ||||
| } | ||||
|  | ||||
| static int encode_data_gcr(uint8_t gcr) | ||||
| { | ||||
|     switch (gcr) | ||||
|     { | ||||
| 		#define GCR_ENTRY(gcr, data) \ | ||||
| 			case data: return gcr; | ||||
| 		#include "data_gcr.h" | ||||
| 		#undef GCR_ENTRY | ||||
|     } | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| /* This is extremely inspired by the MESS implementation, written by Nathan Woods | ||||
|  * and R. Belmont: https://github.com/mamedev/mame/blob/4263a71e64377db11392c458b580c5ae83556bc7/src/lib/formats/ap_dsk35.cpp | ||||
|  */ | ||||
| static Bytes encode_crazy_data(const Bytes& input) | ||||
| { | ||||
|     Bytes output; | ||||
|     ByteWriter bw(output); | ||||
|     ByteReader br(input); | ||||
|  | ||||
| 	uint8_t w1, w2, w3, w4; | ||||
|  | ||||
|     static const int LOOKUP_LEN = MAC_SECTOR_LENGTH / 3; | ||||
|  | ||||
|     uint8_t b1[LOOKUP_LEN + 1]; | ||||
|     uint8_t b2[LOOKUP_LEN + 1]; | ||||
|     uint8_t b3[LOOKUP_LEN + 1]; | ||||
|  | ||||
| 	uint32_t c1 = 0; | ||||
| 	uint32_t c2 = 0; | ||||
| 	uint32_t c3 = 0; | ||||
| 	for (int j=0;; j++) | ||||
| 	{ | ||||
| 		c1 = (c1 & 0xff) << 1; | ||||
| 		if (c1 & 0x0100) | ||||
| 			c1++; | ||||
|  | ||||
| 		uint8_t val = br.read_8(); | ||||
| 		c3 += val; | ||||
| 		if (c1 & 0x0100) | ||||
| 		{ | ||||
| 			c3++; | ||||
| 			c1 &= 0xff; | ||||
| 		} | ||||
| 		b1[j] = (val ^ c1) & 0xff; | ||||
|  | ||||
| 		val = br.read_8(); | ||||
| 		c2 += val; | ||||
| 		if (c3 > 0xff) | ||||
| 		{ | ||||
| 			c2++; | ||||
| 			c3 &= 0xff; | ||||
| 		} | ||||
| 		b2[j] = (val ^ c3) & 0xff; | ||||
|  | ||||
| 		if (br.pos == 524) | ||||
| 			break; | ||||
|  | ||||
| 		val = br.read_8(); | ||||
| 		c1 += val; | ||||
| 		if (c2 > 0xff) | ||||
| 		{ | ||||
| 			c1++; | ||||
| 			c2 &= 0xff; | ||||
| 		} | ||||
| 		b3[j] = (val ^ c2) & 0xff; | ||||
| 	} | ||||
| 	uint32_t c4 = ((c1 & 0xc0) >> 6) | ((c2 & 0xc0) >> 4) | ((c3 & 0xc0) >> 2); | ||||
| 	b3[LOOKUP_LEN] = 0; | ||||
|  | ||||
| 	for (int i = 0; i <= LOOKUP_LEN; i++) | ||||
| 	{ | ||||
| 		w1 = b1[i] & 0x3f; | ||||
| 		w2 = b2[i] & 0x3f; | ||||
| 		w3 = b3[i] & 0x3f; | ||||
| 		w4 =  ((b1[i] & 0xc0) >> 2); | ||||
| 		w4 |= ((b2[i] & 0xc0) >> 4); | ||||
| 		w4 |= ((b3[i] & 0xc0) >> 6); | ||||
|  | ||||
| 		bw.write_8(w4); | ||||
| 		bw.write_8(w1); | ||||
| 		bw.write_8(w2); | ||||
|  | ||||
| 		if (i != LOOKUP_LEN) | ||||
| 			bw.write_8(w3); | ||||
| 	} | ||||
|  | ||||
| 	bw.write_8(c4 & 0x3f); | ||||
| 	bw.write_8(c3 & 0x3f); | ||||
| 	bw.write_8(c2 & 0x3f); | ||||
| 	bw.write_8(c1 & 0x3f); | ||||
|  | ||||
| 	return output; | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, const std::vector<bool>& src) | ||||
| { | ||||
| 	for (bool bit : src) | ||||
| 	{ | ||||
| 		if (cursor < bits.size()) | ||||
| 			bits[cursor++] = bit; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, uint64_t data, int width) | ||||
| { | ||||
| 	cursor += width; | ||||
| 	for (int i=0; i<width; i++) | ||||
| 	{ | ||||
| 		unsigned pos = cursor - i - 1; | ||||
| 		if (pos < bits.size()) | ||||
| 			bits[pos] = data & 1; | ||||
| 		data >>= 1; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static uint8_t encode_side(uint8_t track, uint8_t side) | ||||
| { | ||||
|     /* Mac disks, being weird, use the side byte to encode both the side (in | ||||
|      * bit 5) and also whether we're above track 0x3f (in bit 0). | ||||
|      */ | ||||
|  | ||||
| 	return (side ? 0x20 : 0x00) | ((track>0x3f) ? 0x01 : 0x00); | ||||
| } | ||||
|  | ||||
| static void write_sector(std::vector<bool>& bits, unsigned& cursor, const std::shared_ptr<Sector>& sector) | ||||
| { | ||||
| 	if ((sector->data.size() != 512) && (sector->data.size() != 524)) | ||||
| 		Error() << "unsupported sector size --- you must pick 512 or 524"; | ||||
|  | ||||
| 	write_bits(bits, cursor, 0xff, 1*8); /* pad byte */ | ||||
| 	for (int i=0; i<7; i++) | ||||
| 		write_bits(bits, cursor, 0xff3fcff3fcffLL, 6*8); /* sync */ | ||||
| 	write_bits(bits, cursor, MAC_SECTOR_RECORD, 3*8); | ||||
|  | ||||
|     uint8_t encodedTrack = sector->logicalTrack & 0x3f; | ||||
| 	uint8_t encodedSector = sector->logicalSector; | ||||
| 	uint8_t encodedSide = encode_side(sector->logicalTrack, sector->logicalSide); | ||||
| 	uint8_t formatByte = MAC_FORMAT_BYTE; | ||||
| 	uint8_t headerChecksum = (encodedTrack ^ encodedSector ^ encodedSide ^ formatByte) & 0x3f; | ||||
|  | ||||
| 	write_bits(bits, cursor, encode_data_gcr(encodedTrack), 1*8); | ||||
| 	write_bits(bits, cursor, encode_data_gcr(encodedSector), 1*8); | ||||
| 	write_bits(bits, cursor, encode_data_gcr(encodedSide), 1*8); | ||||
| 	write_bits(bits, cursor, encode_data_gcr(formatByte), 1*8); | ||||
| 	write_bits(bits, cursor, encode_data_gcr(headerChecksum), 1*8); | ||||
|  | ||||
| 	write_bits(bits, cursor, 0xdeaaff, 3*8); | ||||
| 	write_bits(bits, cursor, 0xff3fcff3fcffLL, 6*8); /* sync */ | ||||
| 	write_bits(bits, cursor, MAC_DATA_RECORD, 3*8); | ||||
| 	write_bits(bits, cursor, encode_data_gcr(sector->logicalSector), 1*8); | ||||
|  | ||||
| 	Bytes wireData; | ||||
| 	wireData.writer().append(sector->data.slice(512, 12)).append(sector->data.slice(0, 512)); | ||||
| 	for (uint8_t b : encode_crazy_data(wireData)) | ||||
| 		write_bits(bits, cursor, encode_data_gcr(b), 1*8); | ||||
|  | ||||
| 	write_bits(bits, cursor, 0xdeaaff, 3*8); | ||||
| } | ||||
|  | ||||
| class MacintoshEncoder : public AbstractEncoder | ||||
| { | ||||
| public: | ||||
| 	MacintoshEncoder(const EncoderProto& config): | ||||
| 		AbstractEncoder(config), | ||||
| 		_config(config.macintosh()) | ||||
| 	{} | ||||
|  | ||||
| public: | ||||
| 	std::vector<std::shared_ptr<Sector>> collectSectors(int physicalTrack, int physicalSide, const Image& image) override | ||||
| 	{ | ||||
| 		std::vector<std::shared_ptr<Sector>> sectors; | ||||
|  | ||||
| 		if ((physicalTrack >= 0) && (physicalTrack < MAC_TRACKS_PER_DISK)) | ||||
|         { | ||||
|             unsigned numSectors = sectorsForTrack(physicalTrack); | ||||
|             for (int sectorId=0; sectorId<numSectors; sectorId++) | ||||
|             { | ||||
|                 const auto& sector = image.get(physicalTrack, physicalSide, sectorId); | ||||
|                 if (sector) | ||||
|                     sectors.push_back(sector); | ||||
|             } | ||||
|         } | ||||
|  | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
|     std::unique_ptr<Fluxmap> encode(int physicalTrack, int physicalSide, | ||||
| 			const std::vector<std::shared_ptr<Sector>>& sectors, const Image& image) override | ||||
| 	{ | ||||
| 		if ((physicalTrack < 0) || (physicalTrack >= MAC_TRACKS_PER_DISK)) | ||||
| 			return std::unique_ptr<Fluxmap>(); | ||||
|  | ||||
| 		double clockRateUs = clockRateUsForTrack(physicalTrack) * _config.clock_compensation_factor(); | ||||
| 		int bitsPerRevolution = 200000.0 / clockRateUs; | ||||
| 		std::vector<bool> bits(bitsPerRevolution); | ||||
| 		unsigned cursor = 0; | ||||
|  | ||||
| 		fillBitmapTo(bits, cursor, _config.post_index_gap_us() / clockRateUs, { true, false }); | ||||
| 		lastBit = false; | ||||
|  | ||||
| 		for (const auto& sector : sectors) | ||||
| 			write_sector(bits, cursor, sector); | ||||
|  | ||||
| 		if (cursor >= bits.size()) | ||||
| 			Error() << fmt::format("track data overrun by {} bits", cursor - bits.size()); | ||||
| 		fillBitmapTo(bits, cursor, bits.size(), { true, false }); | ||||
|  | ||||
| 		std::unique_ptr<Fluxmap> fluxmap(new Fluxmap); | ||||
| 		fluxmap->appendBits(bits, clockRateUs*1e3); | ||||
| 		return fluxmap; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const MacintoshEncoderProto& _config; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractEncoder> createMacintoshEncoder(const EncoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractEncoder>(new MacintoshEncoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										22
									
								
								arch/macintosh/macintosh.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								arch/macintosh/macintosh.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| #ifndef MACINTOSH_H | ||||
| #define MACINTOSH_H | ||||
|  | ||||
| #define MAC_SECTOR_RECORD   0xd5aa96 /* 1101 0101 1010 1010 1001 0110 */ | ||||
| #define MAC_DATA_RECORD     0xd5aaad /* 1101 0101 1010 1010 1010 1101 */ | ||||
|  | ||||
| #define MAC_SECTOR_LENGTH   524 /* yes, really */ | ||||
| #define MAC_ENCODED_SECTOR_LENGTH 703 | ||||
| #define MAC_FORMAT_BYTE     0x22 | ||||
|  | ||||
| #define MAC_TRACKS_PER_DISK 80 | ||||
|  | ||||
| class AbstractEncoder; | ||||
| class AbstractDecoder; | ||||
| class DecoderProto; | ||||
| class EncoderProto; | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createMacintoshDecoder(const DecoderProto& config); | ||||
| extern std::unique_ptr<AbstractEncoder> createMacintoshEncoder(const EncoderProto& config); | ||||
|  | ||||
| #endif | ||||
|  | ||||
							
								
								
									
										14
									
								
								arch/macintosh/macintosh.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								arch/macintosh/macintosh.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| import "lib/common.proto"; | ||||
|  | ||||
| message MacintoshDecoderProto {} | ||||
|  | ||||
| message MacintoshEncoderProto { | ||||
| 	optional double post_index_gap_us = 1 [default = 0.0, | ||||
| 		(help) = "post-index gap before first sector header (microseconds)."]; | ||||
|  | ||||
| 	optional double clock_compensation_factor = 2 [default = 1.0, | ||||
| 		(help) = "scale the output clock by this much."]; | ||||
| } | ||||
|  | ||||
							
								
								
									
										75
									
								
								arch/micropolis/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										75
									
								
								arch/micropolis/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,75 @@ | ||||
| #include "globals.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "micropolis.h" | ||||
| #include "bytes.h" | ||||
| #include "fmt/format.h" | ||||
|  | ||||
| /* The sector has a preamble of MFM 0x00s and uses 0xFF as a sync pattern. */ | ||||
| static const FluxPattern SECTOR_SYNC_PATTERN(32, 0xaaaa5555); | ||||
|  | ||||
| /* Adds all bytes, with carry. */ | ||||
| uint8_t micropolisChecksum(const Bytes& bytes) { | ||||
| 	ByteReader br(bytes); | ||||
| 	uint16_t sum = 0; | ||||
| 	while (!br.eof()) { | ||||
| 		if (sum > 0xFF) { | ||||
| 			sum -= 0x100 - 1; | ||||
| 		} | ||||
| 		sum += br.read_8(); | ||||
| 	} | ||||
| 	/* The last carry is ignored */ | ||||
| 	return sum & 0xFF; | ||||
| } | ||||
|  | ||||
| class MicropolisDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	MicropolisDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
|  | ||||
| 	RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		_fmr->seekToIndexMark(); | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(SECTOR_SYNC_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_SYNC_PATTERN) { | ||||
| 			return SECTOR_RECORD; | ||||
| 		} | ||||
| 		return UNKNOWN_RECORD; | ||||
| 	} | ||||
|  | ||||
| 	void decodeSectorRecord() | ||||
| 	{ | ||||
| 		readRawBits(16); | ||||
| 		auto rawbits = readRawBits(MICROPOLIS_ENCODED_SECTOR_SIZE*16); | ||||
| 		auto bytes = decodeFmMfm(rawbits).slice(0, MICROPOLIS_ENCODED_SECTOR_SIZE); | ||||
| 		ByteReader br(bytes); | ||||
|  | ||||
| 		br.read_8();  /* sync */ | ||||
| 		_sector->logicalTrack = br.read_8(); | ||||
| 		_sector->logicalSide = _sector->physicalHead; | ||||
| 		_sector->logicalSector = br.read_8(); | ||||
| 		if (_sector->logicalSector > 15) | ||||
| 			return; | ||||
| 		if (_sector->logicalTrack > 77) | ||||
| 			return; | ||||
|  | ||||
| 		br.read(10);  /* OS data or padding */ | ||||
| 		_sector->data = br.read(256); | ||||
| 		uint8_t wantChecksum = br.read_8(); | ||||
| 		uint8_t gotChecksum = micropolisChecksum(bytes.slice(1, 2+266)); | ||||
| 		br.read(5);  /* 4 byte ECC and ECC-present flag */ | ||||
|  | ||||
| 		_sector->status = (wantChecksum == gotChecksum) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createMicropolisDecoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new MicropolisDecoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										113
									
								
								arch/micropolis/encoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										113
									
								
								arch/micropolis/encoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,113 @@ | ||||
| #include "globals.h" | ||||
| #include "micropolis.h" | ||||
| #include "sector.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "image.h" | ||||
| #include "lib/encoders/encoders.pb.h" | ||||
|  | ||||
| static void write_sector(std::vector<bool>& bits, unsigned& cursor, const std::shared_ptr<Sector>& sector) | ||||
| { | ||||
| 	if ((sector->data.size() != 256) && (sector->data.size() != MICROPOLIS_ENCODED_SECTOR_SIZE)) | ||||
| 		Error() << "unsupported sector size --- you must pick 256 or 275"; | ||||
|  | ||||
| 	int fullSectorSize = 40 + MICROPOLIS_ENCODED_SECTOR_SIZE + 40 + 35; | ||||
| 	auto fullSector = std::make_shared<std::vector<uint8_t>>(); | ||||
| 	fullSector->reserve(fullSectorSize); | ||||
| 	/* sector preamble */ | ||||
| 	for (int i=0; i<40; i++) | ||||
| 		fullSector->push_back(0); | ||||
| 	Bytes sectorData; | ||||
| 	if (sector->data.size() == MICROPOLIS_ENCODED_SECTOR_SIZE) | ||||
| 		sectorData = sector->data; | ||||
| 	else | ||||
| 	{ | ||||
| 		ByteWriter writer(sectorData); | ||||
| 		writer.write_8(0xff); /* Sync */ | ||||
| 		writer.write_8(sector->logicalTrack); | ||||
| 		writer.write_8(sector->logicalSector); | ||||
| 		for (int i=0; i<10; i++) | ||||
| 			writer.write_8(0); /* Padding */ | ||||
| 		writer += sector->data; | ||||
| 		writer.write_8(micropolisChecksum(sectorData.slice(1))); | ||||
| 		for (int i=0; i<5; i++) | ||||
| 			writer.write_8(0); /* 4 byte ECC and ECC not present flag */ | ||||
| 	} | ||||
| 	for (uint8_t b : sectorData) | ||||
| 		fullSector->push_back(b); | ||||
| 	/* sector postamble */ | ||||
| 	for (int i=0; i<40; i++) | ||||
| 		fullSector->push_back(0); | ||||
| 	/* filler */ | ||||
| 	for (int i=0; i<35; i++) | ||||
| 		fullSector->push_back(0); | ||||
|  | ||||
| 	if (fullSector->size() != fullSectorSize) | ||||
| 		Error() << "sector mismatched length"; | ||||
| 	bool lastBit = false; | ||||
| 	encodeMfm(bits, cursor, fullSector, lastBit); | ||||
| 	/* filler */ | ||||
| 	for (int i=0; i<5; i++) | ||||
| 	{ | ||||
| 		bits[cursor++] = 1; | ||||
| 		bits[cursor++] = 0; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| class MicropolisEncoder : public AbstractEncoder | ||||
| { | ||||
| public: | ||||
| 	MicropolisEncoder(const EncoderProto& config): | ||||
| 		AbstractEncoder(config), | ||||
| 		_config(config.micropolis()) | ||||
| 	{} | ||||
|  | ||||
| 	std::vector<std::shared_ptr<Sector>> collectSectors(int physicalTrack, int physicalSide, const Image& image) override | ||||
| 	{ | ||||
| 		std::vector<std::shared_ptr<Sector>> sectors; | ||||
|  | ||||
| 		if ((physicalTrack >= 0) && (physicalTrack < 77)) | ||||
| 		{ | ||||
| 			for (int sectorId = 0; sectorId < 16; sectorId++) | ||||
| 			{ | ||||
| 				const auto& sector = image.get(physicalTrack, physicalSide, sectorId); | ||||
| 				if (sector) | ||||
| 					sectors.push_back(sector); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
| 	std::unique_ptr<Fluxmap> encode(int physicalTrack, int physicalSide, | ||||
| 			const std::vector<std::shared_ptr<Sector>>& sectors, const Image& image) override | ||||
| 	{ | ||||
| 		int bitsPerRevolution = 100000; | ||||
| 		double clockRateUs = 2.00; | ||||
|  | ||||
| 		if ((physicalTrack < 0) || (physicalTrack >= 77) || sectors.empty()) | ||||
| 			return std::unique_ptr<Fluxmap>(); | ||||
|  | ||||
| 		std::vector<bool> bits(bitsPerRevolution); | ||||
| 		unsigned cursor = 0; | ||||
|  | ||||
| 		for (const auto& sectorData : sectors) | ||||
| 			write_sector(bits, cursor, sectorData); | ||||
|  | ||||
| 		if (cursor != bits.size()) | ||||
| 			Error() << "track data mismatched length"; | ||||
|  | ||||
| 		std::unique_ptr<Fluxmap> fluxmap(new Fluxmap); | ||||
| 		fluxmap->appendBits(bits, clockRateUs * 1e3); | ||||
| 		return fluxmap; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const MicropolisEncoderProto& _config; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractEncoder> createMicropolisEncoder(const EncoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractEncoder>(new MicropolisEncoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										16
									
								
								arch/micropolis/micropolis.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								arch/micropolis/micropolis.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | ||||
| #ifndef MICROPOLIS_H | ||||
| #define MICROPOLIS_H | ||||
|  | ||||
| #define MICROPOLIS_ENCODED_SECTOR_SIZE (1+2+266+6) | ||||
|  | ||||
| class AbstractDecoder; | ||||
| class AbstractEncoder; | ||||
| class EncoderProto; | ||||
| class DecoderProto; | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createMicropolisDecoder(const DecoderProto& config); | ||||
| extern std::unique_ptr<AbstractEncoder> createMicropolisEncoder(const EncoderProto& config); | ||||
|  | ||||
| extern uint8_t micropolisChecksum(const Bytes& bytes); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										5
									
								
								arch/micropolis/micropolis.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								arch/micropolis/micropolis.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| message MicropolisDecoderProto {} | ||||
| message MicropolisEncoderProto {} | ||||
|  | ||||
							
								
								
									
										93
									
								
								arch/mx/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								arch/mx/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,93 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "mx/mx.h" | ||||
| #include "crc.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "sector.h" | ||||
| #include <string.h> | ||||
|  | ||||
| const int SECTOR_SIZE = 256; | ||||
|  | ||||
| /* | ||||
|  * MX disks are a bunch of sectors glued together with no gaps or sync markers, | ||||
|  * following a single beginning-of-track synchronisation and identification | ||||
|  * sequence. | ||||
|  */ | ||||
|  | ||||
| /* FM beginning of track marker: | ||||
|  * 1010 1010 1010 1010 1111 1111 1010 1111 | ||||
|  *    a    a    a    a    f    f    a    f | ||||
|  */ | ||||
| const FluxPattern ID_PATTERN(32, 0xaaaaffaf); | ||||
|  | ||||
| class MxDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	MxDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
|  | ||||
|     void beginTrack() | ||||
| 	{ | ||||
| 		_currentSector = -1; | ||||
| 		_clock = 0; | ||||
| 	} | ||||
|  | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		if (_currentSector == -1) | ||||
| 		{ | ||||
| 			/* First sector in the track: look for the sync marker. */ | ||||
| 			const FluxMatcher* matcher = nullptr; | ||||
| 			_sector->clock = _clock = _fmr->seekToPattern(ID_PATTERN, matcher); | ||||
| 			readRawBits(32); /* skip the ID mark */ | ||||
| 			_logicalTrack = decodeFmMfm(readRawBits(32)).slice(0, 32).reader().read_be16(); | ||||
| 		} | ||||
| 		else if (_currentSector == 10) | ||||
| 		{ | ||||
| 			/* That was the last sector on the disk. */ | ||||
| 			return UNKNOWN_RECORD; | ||||
| 		} | ||||
| 		else | ||||
| 		{ | ||||
| 			/* Otherwise we assume the clock from the first sector is still valid. | ||||
| 			 * The decoder framwork will automatically stop when we hit the end of | ||||
| 			 * the track. */ | ||||
| 			_sector->clock = _clock; | ||||
| 		} | ||||
|  | ||||
| 		_currentSector++; | ||||
| 		return SECTOR_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		auto bits = readRawBits((SECTOR_SIZE+2)*16); | ||||
| 		auto bytes = decodeFmMfm(bits).slice(0, SECTOR_SIZE+2).swab(); | ||||
|  | ||||
| 		uint16_t gotChecksum = 0; | ||||
| 		ByteReader br(bytes); | ||||
| 		for (int i=0; i<(SECTOR_SIZE/2); i++) | ||||
| 			gotChecksum += br.read_le16(); | ||||
| 		uint16_t wantChecksum = br.read_le16(); | ||||
|  | ||||
| 		_sector->logicalTrack = _logicalTrack; | ||||
| 		_sector->logicalSide = _sector->physicalHead; | ||||
| 		_sector->logicalSector = _currentSector; | ||||
| 		_sector->data = bytes.slice(0, SECTOR_SIZE); | ||||
| 		_sector->status = (gotChecksum == wantChecksum) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
|     nanoseconds_t _clock; | ||||
|     int _currentSector; | ||||
|     int _logicalTrack; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createMxDecoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new MxDecoder(config)); | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										8
									
								
								arch/mx/mx.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								arch/mx/mx.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| #ifndef MX_H | ||||
| #define MX_H | ||||
|  | ||||
| #include "decoders/decoders.h" | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createMxDecoder(const DecoderProto& config); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										4
									
								
								arch/mx/mx.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								arch/mx/mx.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| message MxDecoderProto {} | ||||
|  | ||||
							
								
								
									
										196
									
								
								arch/northstar/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										196
									
								
								arch/northstar/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,196 @@ | ||||
| /* Decoder for North Star 10-sector hard-sectored disks. | ||||
|  * | ||||
|  * Supports both single- and double-density.  For the sector format and | ||||
|  * checksum algorithm, see pp. 33 of the North Star Double Density Controller | ||||
|  * manual: | ||||
|  * | ||||
|  * http://bitsavers.org/pdf/northstar/boards/Northstar_MDS-A-D_1978.pdf | ||||
|  * | ||||
|  * North Star disks do not contain any track/head/sector information | ||||
|  * encoded in the sector record.  For this reason, we have to be absolutely | ||||
|  * sure that the hardSectorId is correct. | ||||
|  */ | ||||
|  | ||||
| #include "globals.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "northstar.h" | ||||
| #include "bytes.h" | ||||
| #include "lib/decoders/decoders.pb.h" | ||||
| #include "fmt/format.h" | ||||
|  | ||||
| /* | ||||
|  * MFM sectors have 32 bytes of 00's followed by two sync characters, | ||||
|  * specified in the North Star MDS manual as 0xFBFB. | ||||
|  * | ||||
|  * This is true for most disks; however, I found a few disks, including an | ||||
|  * original North Star DOS/BASIC v2.2.1 DQ disk) that uses 0xFBnn, where | ||||
|  * nn is an incrementing pattern. | ||||
|  * | ||||
|  * 00        00        00        F         B | ||||
|  * 0000 0000 0000 0000 0000 0000 0101 0101 0100 0101 | ||||
|  * A    A    A    A    A    A    5    5    4    5 | ||||
|  */ | ||||
| static const FluxPattern MFM_PATTERN(64, 0xAAAAAAAAAAAA5545LL); | ||||
|  | ||||
| /* FM sectors have 16 bytes of 00's followed by 0xFB. | ||||
|  * 00        FB | ||||
|  * 0000 0000 1111 1111 1110 1111 | ||||
|  * A    A    F    F    E    F | ||||
|  */ | ||||
| static const FluxPattern FM_PATTERN(64, 0xAAAAAAAAAAAAFFEFLL); | ||||
|  | ||||
| const FluxMatchers ANY_SECTOR_PATTERN( | ||||
| 	{ | ||||
| 		&MFM_PATTERN, | ||||
| 		&FM_PATTERN, | ||||
| 	} | ||||
| ); | ||||
|  | ||||
| /* Checksum is initially 0. | ||||
|  * For each data byte, XOR with the current checksum. | ||||
|  * Rotate checksum left, carrying bit 7 to bit 0. | ||||
|  */ | ||||
| uint8_t northstarChecksum(const Bytes& bytes) { | ||||
| 	ByteReader br(bytes); | ||||
| 	uint8_t checksum = 0; | ||||
|  | ||||
| 	while (!br.eof()) { | ||||
| 		checksum ^= br.read_8(); | ||||
| 		checksum = ((checksum << 1) | ((checksum >> 7))); | ||||
| 	} | ||||
|  | ||||
| 	return checksum; | ||||
| } | ||||
|  | ||||
| class NorthstarDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	NorthstarDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config), | ||||
| 		_config(config.northstar()) | ||||
| 	{} | ||||
|  | ||||
| 	/* Search for FM or MFM sector record */ | ||||
| 	RecordType advanceToNextRecord() override | ||||
| 	{ | ||||
| 		nanoseconds_t now = _fmr->tell().ns(); | ||||
|  | ||||
| 		/* For all but the first sector, seek to the next sector pulse. | ||||
| 		 * The first sector does not contain the sector pulse in the fluxmap. | ||||
| 		 */ | ||||
| 		if (now != 0) { | ||||
| 			_fmr->seekToIndexMark(); | ||||
| 			now = _fmr->tell().ns(); | ||||
| 		} | ||||
|  | ||||
| 		/* Discard a possible partial sector at the end of the track. | ||||
| 		 * This partial sector could be mistaken for a conflicted sector, if | ||||
| 		 * whatever data read happens to match the checksum of 0, which is | ||||
| 		 * rare, but has been observed on some disks. | ||||
| 		 */ | ||||
| 		if (now > (_fmr->getDuration() - 21e6)) { | ||||
| 			_fmr->seekToIndexMark(); | ||||
| 			return(UNKNOWN_RECORD); | ||||
| 		} | ||||
|  | ||||
| 		int msSinceIndex = std::round(now / 1e6); | ||||
|  | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
|  | ||||
| 		/* Note that the seekToPattern ignores the sector pulses, so if | ||||
| 		 * a sector is not found for some reason, the seek will advance | ||||
| 		 * past one or more sector pulses.  For this reason, calculate | ||||
| 		 * _hardSectorId after the sector header is found. | ||||
| 		 */ | ||||
| 		_sector->clock = _fmr->seekToPattern(ANY_SECTOR_PATTERN, matcher); | ||||
|  | ||||
| 		int sectorFoundTimeRaw = std::round((_fmr->tell().ns()) / 1e6); | ||||
| 		int sectorFoundTime; | ||||
|  | ||||
| 		/* Round time to the nearest 20ms */ | ||||
| 		if ((sectorFoundTimeRaw % 20) < 10) { | ||||
| 			sectorFoundTime = (sectorFoundTimeRaw / 20) * 20; | ||||
| 		} | ||||
| 		else { | ||||
| 			sectorFoundTime = ((sectorFoundTimeRaw + 20) / 20) * 20; | ||||
| 		} | ||||
|  | ||||
| 		/* Calculate the sector ID based on time since the index */ | ||||
| 		_hardSectorId = (sectorFoundTime / 20) % 10; | ||||
|  | ||||
| 	//	std::cout << fmt::format( | ||||
| 	//		"Sector ID {}: hole at {}ms, sector start at {}ms", | ||||
| 	//		_hardSectorId, msSinceIndex, sectorFoundTimeRaw) << std::endl; | ||||
|  | ||||
| 		if (matcher == &MFM_PATTERN) { | ||||
| 			_sectorType = SECTOR_TYPE_MFM; | ||||
| 			return SECTOR_RECORD; | ||||
| 		} | ||||
|  | ||||
| 		if (matcher == &FM_PATTERN) { | ||||
| 			_sectorType = SECTOR_TYPE_FM; | ||||
| 			return SECTOR_RECORD; | ||||
| 		} | ||||
|  | ||||
| 		return UNKNOWN_RECORD; | ||||
| 	} | ||||
|  | ||||
| 	void decodeSectorRecord() override | ||||
| 	{ | ||||
| 		unsigned recordSize, payloadSize, headerSize; | ||||
|  | ||||
| 		if (_sectorType == SECTOR_TYPE_MFM) { | ||||
| 			recordSize = NORTHSTAR_ENCODED_SECTOR_SIZE_DD; | ||||
| 			payloadSize = NORTHSTAR_PAYLOAD_SIZE_DD; | ||||
| 			headerSize = NORTHSTAR_HEADER_SIZE_DD; | ||||
| 		} | ||||
| 		else { | ||||
| 			recordSize = NORTHSTAR_ENCODED_SECTOR_SIZE_SD; | ||||
| 			payloadSize = NORTHSTAR_PAYLOAD_SIZE_SD; | ||||
| 			headerSize = NORTHSTAR_HEADER_SIZE_SD; | ||||
| 		} | ||||
|  | ||||
| 		readRawBits(48); | ||||
|  | ||||
| 		auto rawbits = readRawBits(recordSize * 16); | ||||
| 		auto bytes = decodeFmMfm(rawbits).slice(0, recordSize); | ||||
| 		ByteReader br(bytes); | ||||
| 		uint8_t sync_char; | ||||
|  | ||||
| 		_sector->logicalSide = _sector->physicalHead; | ||||
| 		_sector->logicalSector = _hardSectorId; | ||||
| 		_sector->logicalTrack = _sector->physicalCylinder; | ||||
|  | ||||
| 		sync_char = br.read_8();	/* Sync char: 0xFB */ | ||||
| 		if (_sectorType == SECTOR_TYPE_MFM) { | ||||
| 			sync_char = br.read_8();/* MFM second Sync char, usually 0xFB */ | ||||
| 		} | ||||
|  | ||||
| 		_sector->data = br.read(payloadSize); | ||||
|  | ||||
| 		uint8_t wantChecksum = br.read_8(); | ||||
| 		uint8_t gotChecksum = northstarChecksum(bytes.slice(headerSize, payloadSize)); | ||||
|  | ||||
| 		_sector->status = (wantChecksum == gotChecksum) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
|  | ||||
| 	std::set<unsigned> requiredSectors(unsigned cylinder, unsigned head) const override | ||||
| 	{ | ||||
| 		static std::set<unsigned> sectors = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 }; | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const NorthstarDecoderProto& _config; | ||||
| 	uint8_t _sectorType = SECTOR_TYPE_MFM; | ||||
| 	uint8_t _hardSectorId; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createNorthstarDecoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new NorthstarDecoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										166
									
								
								arch/northstar/encoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								arch/northstar/encoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,166 @@ | ||||
| #include "globals.h" | ||||
| #include "northstar.h" | ||||
| #include "sector.h" | ||||
| #include "bytes.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "image.h" | ||||
| #include "lib/encoders/encoders.pb.h" | ||||
|  | ||||
| #define GAP_FILL_SIZE_SD 30 | ||||
| #define PRE_HEADER_GAP_FILL_SIZE_SD 9 | ||||
| #define GAP_FILL_SIZE_DD 62 | ||||
| #define PRE_HEADER_GAP_FILL_SIZE_DD 16 | ||||
|  | ||||
| #define GAP1_FILL_BYTE	(0x4F) | ||||
| #define GAP2_FILL_BYTE	(0x4F) | ||||
|  | ||||
| #define TOTAL_SECTOR_BYTES () | ||||
|  | ||||
| static void write_sector(std::vector<bool>& bits, unsigned& cursor, const std::shared_ptr<Sector>& sector) | ||||
| { | ||||
| 	int preambleSize = 0; | ||||
| 	int encodedSectorSize = 0; | ||||
| 	int gapFillSize = 0; | ||||
| 	int preHeaderGapFillSize = 0; | ||||
|  | ||||
| 	bool doubleDensity; | ||||
|  | ||||
| 	switch (sector->data.size()) { | ||||
| 	case NORTHSTAR_PAYLOAD_SIZE_SD: | ||||
| 		preambleSize = NORTHSTAR_PREAMBLE_SIZE_SD; | ||||
| 		encodedSectorSize = PRE_HEADER_GAP_FILL_SIZE_SD + NORTHSTAR_ENCODED_SECTOR_SIZE_SD + GAP_FILL_SIZE_SD; | ||||
| 		gapFillSize = GAP_FILL_SIZE_SD; | ||||
| 		preHeaderGapFillSize = PRE_HEADER_GAP_FILL_SIZE_SD; | ||||
| 		doubleDensity = false; | ||||
| 		break; | ||||
| 	case NORTHSTAR_PAYLOAD_SIZE_DD: | ||||
| 		preambleSize = NORTHSTAR_PREAMBLE_SIZE_DD; | ||||
| 		encodedSectorSize = PRE_HEADER_GAP_FILL_SIZE_DD + NORTHSTAR_ENCODED_SECTOR_SIZE_DD + GAP_FILL_SIZE_DD; | ||||
| 		gapFillSize = GAP_FILL_SIZE_DD; | ||||
| 		preHeaderGapFillSize = PRE_HEADER_GAP_FILL_SIZE_DD; | ||||
| 		doubleDensity = true; | ||||
| 		break; | ||||
| 	default: | ||||
| 		Error() << "unsupported sector size --- you must pick 256 or 512"; | ||||
| 		break; | ||||
| 	} | ||||
|  | ||||
| 	int fullSectorSize = preambleSize + encodedSectorSize; | ||||
| 	auto fullSector = std::make_shared<std::vector<uint8_t>>(); | ||||
| 	fullSector->reserve(fullSectorSize); | ||||
|  | ||||
| 	/* sector gap after index pulse */ | ||||
| 	for (int i = 0; i < preHeaderGapFillSize; i++) | ||||
| 		fullSector->push_back(GAP1_FILL_BYTE); | ||||
|  | ||||
| 	/* sector preamble */ | ||||
| 	for (int i = 0; i < preambleSize; i++) | ||||
| 		fullSector->push_back(0); | ||||
|  | ||||
| 	Bytes sectorData; | ||||
| 	if (sector->data.size() == encodedSectorSize) | ||||
| 		sectorData = sector->data; | ||||
| 	else { | ||||
| 		ByteWriter writer(sectorData); | ||||
| 		writer.write_8(0xFB);     /* sync character */ | ||||
| 		if (doubleDensity == true) { | ||||
| 			writer.write_8(0xFB); /* Double-density has two sync characters */ | ||||
| 		} | ||||
| 		writer += sector->data; | ||||
| 		if (doubleDensity == true) { | ||||
| 			writer.write_8(northstarChecksum(sectorData.slice(2))); | ||||
| 		} else { | ||||
| 			writer.write_8(northstarChecksum(sectorData.slice(1))); | ||||
| 		} | ||||
| 	} | ||||
| 	for (uint8_t b : sectorData) | ||||
| 		fullSector->push_back(b); | ||||
|  | ||||
| 	if (sector->logicalSector != 9) { | ||||
| 		/* sector postamble */ | ||||
| 		for (int i = 0; i < gapFillSize; i++) | ||||
| 			fullSector->push_back(GAP2_FILL_BYTE); | ||||
|  | ||||
| 		if (fullSector->size() != fullSectorSize) | ||||
| 			Error() << "sector mismatched length (" << sector->data.size() << ") expected: " << fullSector->size() << " got " << fullSectorSize; | ||||
| 	} else { | ||||
| 		/* sector postamble */ | ||||
| 		for (int i = 0; i < gapFillSize; i++) | ||||
| 			fullSector->push_back(GAP2_FILL_BYTE); | ||||
| 	} | ||||
|  | ||||
| 	bool lastBit = false; | ||||
|  | ||||
| 	if (doubleDensity == true) { | ||||
| 		encodeMfm(bits, cursor, fullSector, lastBit); | ||||
| 	} | ||||
| 	else { | ||||
| 		encodeFm(bits, cursor, fullSector); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| class NorthstarEncoder : public AbstractEncoder | ||||
| { | ||||
| public: | ||||
| 	NorthstarEncoder(const EncoderProto& config): | ||||
| 		AbstractEncoder(config), | ||||
| 		_config(config.northstar()) | ||||
| 	{} | ||||
|  | ||||
| 	std::vector<std::shared_ptr<Sector>> collectSectors(int physicalTrack, int physicalSide, const Image& image) override | ||||
| 	{ | ||||
| 		std::vector<std::shared_ptr<Sector>> sectors; | ||||
|  | ||||
| 		if ((physicalTrack >= 0) && (physicalTrack < 35)) | ||||
| 		{ | ||||
| 			for (int sectorId = 0; sectorId < 10; sectorId++) | ||||
| 			{ | ||||
| 				const auto& sector = image.get(physicalTrack, physicalSide, sectorId); | ||||
| 				if (sector) | ||||
| 					sectors.push_back(sector); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
| 	std::unique_ptr<Fluxmap> encode(int physicalTrack, int physicalSide, | ||||
| 			const std::vector<std::shared_ptr<Sector>>& sectors, const Image& image) override | ||||
| 	{ | ||||
| 		int bitsPerRevolution = 100000; | ||||
| 		double clockRateUs = 4.00; | ||||
|  | ||||
| 		if ((physicalTrack < 0) || (physicalTrack >= 35) || sectors.empty()) | ||||
| 			return std::unique_ptr<Fluxmap>(); | ||||
|  | ||||
| 		const auto& sector = *sectors.begin(); | ||||
| 		if (sector->data.size() == NORTHSTAR_PAYLOAD_SIZE_SD) { | ||||
| 			bitsPerRevolution /= 2;		// FM | ||||
| 		} else { | ||||
| 			clockRateUs /= 2.00; | ||||
| 		} | ||||
|  | ||||
| 		std::vector<bool> bits(bitsPerRevolution); | ||||
| 		unsigned cursor = 0; | ||||
|  | ||||
| 		for (const auto& sectorData : sectors) | ||||
| 			write_sector(bits, cursor, sectorData); | ||||
|  | ||||
| 		if (cursor > bits.size()) | ||||
| 			Error() << "track data overrun"; | ||||
|  | ||||
| 		std::unique_ptr<Fluxmap> fluxmap(new Fluxmap); | ||||
| 		fluxmap->appendBits(bits, clockRateUs * 1e3); | ||||
| 		return fluxmap; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const NorthstarEncoderProto& _config; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractEncoder> createNorthstarEncoder(const EncoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractEncoder>(new NorthstarEncoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										38
									
								
								arch/northstar/northstar.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										38
									
								
								arch/northstar/northstar.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,38 @@ | ||||
| #ifndef NORTHSTAR_H | ||||
| #define NORTHSTAR_H | ||||
|  | ||||
| /* Northstar floppies are 10-hard sectored disks with a sector format as follows: | ||||
|  * | ||||
|  * |----------------------------------| | ||||
|  * | SYNC Byte  | Payload  | Checksum | | ||||
|  * |------------+----------+----------| | ||||
|  * | 1 (0xFB)   | 256 (SD) |    1     | | ||||
|  * | 2 (0xFBFB) | 512 (DD) |          | | ||||
|  * |----------------------------------| | ||||
|  * | ||||
|  */ | ||||
|  | ||||
| #define NORTHSTAR_PREAMBLE_SIZE_SD		(16) | ||||
| #define NORTHSTAR_PREAMBLE_SIZE_DD		(32) | ||||
| #define NORTHSTAR_HEADER_SIZE_SD		(1) | ||||
| #define NORTHSTAR_HEADER_SIZE_DD		(2) | ||||
| #define NORTHSTAR_PAYLOAD_SIZE_SD		(256) | ||||
| #define NORTHSTAR_PAYLOAD_SIZE_DD		(512) | ||||
| #define NORTHSTAR_CHECKSUM_SIZE		(1) | ||||
| #define NORTHSTAR_ENCODED_SECTOR_SIZE_SD	(NORTHSTAR_HEADER_SIZE_SD + NORTHSTAR_PAYLOAD_SIZE_SD + NORTHSTAR_CHECKSUM_SIZE) | ||||
| #define NORTHSTAR_ENCODED_SECTOR_SIZE_DD	(NORTHSTAR_HEADER_SIZE_DD + NORTHSTAR_PAYLOAD_SIZE_DD + NORTHSTAR_CHECKSUM_SIZE) | ||||
|  | ||||
| #define SECTOR_TYPE_MFM			(0) | ||||
| #define SECTOR_TYPE_FM				(1) | ||||
|  | ||||
| class AbstractDecoder; | ||||
| class AbstractEncoder; | ||||
| class EncoderProto; | ||||
| class DecoderProto; | ||||
|  | ||||
| extern uint8_t northstarChecksum(const Bytes& bytes); | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createNorthstarDecoder(const DecoderProto& config); | ||||
| extern std::unique_ptr<AbstractEncoder> createNorthstarEncoder(const EncoderProto& config); | ||||
|  | ||||
| #endif /* NORTHSTAR */ | ||||
							
								
								
									
										5
									
								
								arch/northstar/northstar.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								arch/northstar/northstar.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| message NorthstarDecoderProto {} | ||||
| message NorthstarEncoderProto {} | ||||
|  | ||||
							
								
								
									
										98
									
								
								arch/tids990/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								arch/tids990/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,98 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "tids990/tids990.h" | ||||
| #include "crc.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "sector.h" | ||||
| #include <string.h> | ||||
| #include <fmt/format.h> | ||||
|  | ||||
| /* The Texas Instruments DS990 uses MFM with a scheme similar to a simplified | ||||
|  * version of the IBM record scheme (it's actually easier to parse than IBM). | ||||
|  * There are 26 sectors per track, each holding a rather weird 288 bytes. | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * Sector record: | ||||
|  * data:    0  1  0  1  0  1  0  1 .0  0  0  0  1  0  1  0  = 0x550a | ||||
|  * mfm:     00 01 00 01 00 01 00 01.00 10 10 10 01 00 01 00 = 0x11112a44 | ||||
|  * special: 00 01 00 01 00 01 00 01.00 10 00 10 01 00 01 00 = 0x11112244 | ||||
|  *                                        ^^ | ||||
|  * When shifted out of phase, the special 0xa1 byte becomes an illegal | ||||
|  * encoding (you can't do 10 00). So this can't be spoofed by user data. | ||||
|  */ | ||||
| const FluxPattern SECTOR_RECORD_PATTERN(32, 0x11112244); | ||||
|  | ||||
| /* | ||||
|  * Data record: | ||||
|  * data:    0  1  0  1  0  1  0  1 .0  0  0  0  1  0  1  1  = 0x550c | ||||
|  * mfm:     00 01 00 01 00 01 00 01.00 10 10 10 01 00 01 01 = 0x11112a45 | ||||
|  * special: 00 01 00 01 00 01 00 01.00 10 00 10 01 00 01 01 = 0x11112245 | ||||
|  *                                        ^^ | ||||
|  * When shifted out of phase, the special 0xa1 byte becomes an illegal | ||||
|  * encoding (you can't do 10 00). So this can't be spoofed by user data. | ||||
|  */ | ||||
| const FluxPattern DATA_RECORD_PATTERN(32, 0x11112245); | ||||
|  | ||||
| const FluxMatchers ANY_RECORD_PATTERN({ &SECTOR_RECORD_PATTERN, &DATA_RECORD_PATTERN }); | ||||
|  | ||||
| class Tids990Decoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	Tids990Decoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
|  | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_RECORD_PATTERN) | ||||
| 			return RecordType::SECTOR_RECORD; | ||||
| 		if (matcher == &DATA_RECORD_PATTERN) | ||||
| 			return RecordType::DATA_RECORD; | ||||
| 		return RecordType::UNKNOWN_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		auto bits = readRawBits(TIDS990_SECTOR_RECORD_SIZE*16); | ||||
| 		auto bytes = decodeFmMfm(bits).slice(0, TIDS990_SECTOR_RECORD_SIZE); | ||||
|  | ||||
| 		ByteReader br(bytes); | ||||
| 		uint16_t gotChecksum = crc16(CCITT_POLY, bytes.slice(1, TIDS990_SECTOR_RECORD_SIZE-3)); | ||||
|  | ||||
| 		br.seek(2); | ||||
| 		_sector->logicalSide = br.read_8() >> 3; | ||||
| 		_sector->logicalTrack = br.read_8(); | ||||
| 		br.read_8(); /* number of sectors per track */ | ||||
| 		_sector->logicalSector = br.read_8(); | ||||
| 		br.read_be16(); /* sector size */ | ||||
| 		uint16_t wantChecksum = br.read_be16(); | ||||
|  | ||||
| 		if (wantChecksum == gotChecksum) | ||||
| 			_sector->status = Sector::DATA_MISSING; /* correct but unintuitive */ | ||||
| 	} | ||||
|  | ||||
| 	void decodeDataRecord() | ||||
| 	{ | ||||
| 		auto bits = readRawBits(TIDS990_DATA_RECORD_SIZE*16); | ||||
| 		auto bytes = decodeFmMfm(bits).slice(0, TIDS990_DATA_RECORD_SIZE); | ||||
|  | ||||
| 		ByteReader br(bytes); | ||||
| 		uint16_t gotChecksum = crc16(CCITT_POLY, bytes.slice(1, TIDS990_DATA_RECORD_SIZE-3)); | ||||
|  | ||||
| 		br.seek(2); | ||||
| 		_sector->data = br.read(TIDS990_PAYLOAD_SIZE); | ||||
| 		uint16_t wantChecksum = br.read_be16(); | ||||
| 		_sector->status = (wantChecksum == gotChecksum) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createTids990Decoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new Tids990Decoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										168
									
								
								arch/tids990/encoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								arch/tids990/encoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,168 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "tids990.h" | ||||
| #include "crc.h" | ||||
| #include "writer.h" | ||||
| #include "image.h" | ||||
| #include "arch/tids990/tids990.pb.h" | ||||
| #include "lib/encoders/encoders.pb.h" | ||||
| #include <fmt/format.h> | ||||
|  | ||||
| static int charToInt(char c) | ||||
| { | ||||
| 	if (isdigit(c)) | ||||
| 		return c - '0'; | ||||
| 	return 10 + tolower(c) - 'a'; | ||||
| } | ||||
|  | ||||
| static uint8_t decodeUint16(uint16_t raw) | ||||
| { | ||||
| 	Bytes b; | ||||
| 	ByteWriter bw(b); | ||||
| 	bw.write_be16(raw); | ||||
| 	return decodeFmMfm(b.toBits())[0]; | ||||
| } | ||||
|  | ||||
| class Tids990Encoder : public AbstractEncoder | ||||
| { | ||||
| public: | ||||
| 	Tids990Encoder(const EncoderProto& config): | ||||
| 		AbstractEncoder(config), | ||||
| 		_config(config.tids990()) | ||||
| 	{} | ||||
|  | ||||
| private: | ||||
| 	void writeRawBits(uint32_t data, int width) | ||||
| 	{ | ||||
| 		_cursor += width; | ||||
| 		_lastBit = data & 1; | ||||
| 		for (int i=0; i<width; i++) | ||||
| 		{ | ||||
| 			unsigned pos = _cursor - i - 1; | ||||
| 			if (pos < _bits.size()) | ||||
| 				_bits[pos] = data & 1; | ||||
| 			data >>= 1; | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| 	void writeBytes(const Bytes& bytes) | ||||
| 	{ | ||||
| 		encodeMfm(_bits, _cursor, bytes, _lastBit); | ||||
| 	} | ||||
|  | ||||
| 	void writeBytes(int count, uint8_t byte) | ||||
| 	{ | ||||
| 		Bytes bytes = { byte }; | ||||
| 		for (int i=0; i<count; i++) | ||||
| 			writeBytes(bytes); | ||||
| 	} | ||||
|  | ||||
| public: | ||||
| 	std::vector<std::shared_ptr<Sector>> collectSectors(int physicalTrack, int physicalSide, const Image& image) override | ||||
| 	{ | ||||
| 		std::vector<std::shared_ptr<Sector>> sectors; | ||||
|  | ||||
| 		for (char sectorChar : _config.sector_skew()) | ||||
|         { | ||||
| 			int sectorId = charToInt(sectorChar); | ||||
| 			const auto& sector = image.get(physicalTrack, physicalSide, sectorId); | ||||
| 			if (sector) | ||||
| 				sectors.push_back(sector); | ||||
|         } | ||||
|  | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
|     std::unique_ptr<Fluxmap> encode(int physicalTrack, int physicalSide, | ||||
| 			const std::vector<std::shared_ptr<Sector>>& sectors, const Image& image) override | ||||
| 	{ | ||||
| 		double clockRateUs = 1e3 / _config.clock_rate_khz() / 2.0; | ||||
| 		int bitsPerRevolution = (_config.track_length_ms() * 1000.0) / clockRateUs; | ||||
| 		_bits.resize(bitsPerRevolution); | ||||
| 		_cursor = 0; | ||||
|  | ||||
| 		uint8_t am1Unencoded = decodeUint16(_config.am1_byte()); | ||||
| 		uint8_t am2Unencoded = decodeUint16(_config.am2_byte()); | ||||
|  | ||||
| 		writeBytes(_config.gap1_bytes(), 0x55); | ||||
|  | ||||
| 		bool first = true; | ||||
| 		for (char sectorChar : _config.sector_skew()) | ||||
| 		{ | ||||
| 			int sectorId = charToInt(sectorChar); | ||||
| 			if (!first) | ||||
| 				writeBytes(_config.gap3_bytes(), 0x55); | ||||
| 			first = false; | ||||
|  | ||||
| 			const auto& sectorData = image.get(physicalTrack, physicalSide, sectorId); | ||||
| 			if (!sectorData) | ||||
| 				Error() << fmt::format("format tried to find sector {} which wasn't in the input file", sectorId); | ||||
|  | ||||
| 			/* Writing the sector and data records are fantastically annoying. | ||||
| 			 * The CRC is calculated from the *very start* of the record, and | ||||
| 			 * include the malformed marker bytes. Our encoder doesn't know | ||||
| 			 * about this, of course, with the result that we have to construct | ||||
| 			 * the unencoded header, calculate the checksum, and then use the | ||||
| 			 * same logic to emit the bytes which require special encoding | ||||
| 			 * before encoding the rest of the header normally. */ | ||||
|  | ||||
| 			{ | ||||
| 				Bytes header; | ||||
| 				ByteWriter bw(header); | ||||
|  | ||||
| 				writeBytes(12, 0x55); | ||||
| 				bw.write_8(am1Unencoded); | ||||
| 				bw.write_8(sectorData->logicalSide << 3); | ||||
| 				bw.write_8(sectorData->logicalTrack); | ||||
| 				bw.write_8(_config.sector_count()); | ||||
| 				bw.write_8(sectorData->logicalSector); | ||||
| 				bw.write_be16(sectorData->data.size()); | ||||
| 				uint16_t crc = crc16(CCITT_POLY, header); | ||||
| 				bw.write_be16(crc); | ||||
|  | ||||
| 				writeRawBits(_config.am1_byte(), 16); | ||||
| 				writeBytes(header.slice(1)); | ||||
| 			} | ||||
|  | ||||
| 			writeBytes(_config.gap2_bytes(), 0x55); | ||||
|  | ||||
| 			{ | ||||
| 				Bytes data; | ||||
| 				ByteWriter bw(data); | ||||
|  | ||||
| 				writeBytes(12, 0x55); | ||||
| 				bw.write_8(am2Unencoded); | ||||
|  | ||||
| 				bw += sectorData->data; | ||||
| 				uint16_t crc = crc16(CCITT_POLY, data); | ||||
| 				bw.write_be16(crc); | ||||
|  | ||||
| 				writeRawBits(_config.am2_byte(), 16); | ||||
| 				writeBytes(data.slice(1)); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		if (_cursor >= _bits.size()) | ||||
| 			Error() << "track data overrun"; | ||||
| 		while (_cursor < _bits.size()) | ||||
| 			writeBytes(1, 0x55); | ||||
| 		 | ||||
| 		std::unique_ptr<Fluxmap> fluxmap(new Fluxmap); | ||||
| 		fluxmap->appendBits(_bits, clockRateUs*1e3); | ||||
| 		return fluxmap; | ||||
| 	} | ||||
|  | ||||
| private: | ||||
| 	const Tids990EncoderProto& _config; | ||||
| 	std::vector<bool> _bits; | ||||
| 	unsigned _cursor; | ||||
| 	bool _lastBit; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractEncoder> createTids990Encoder(const EncoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractEncoder>(new Tids990Encoder(config)); | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										18
									
								
								arch/tids990/tids990.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								arch/tids990/tids990.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | ||||
| #ifndef TIDS990_H | ||||
| #define TIDS990_H | ||||
|  | ||||
| #define TIDS990_PAYLOAD_SIZE       288 /* bytes */ | ||||
| #define TIDS990_SECTOR_RECORD_SIZE 10 /* bytes */ | ||||
| #define TIDS990_DATA_RECORD_SIZE   (TIDS990_PAYLOAD_SIZE + 4) /* bytes */ | ||||
|  | ||||
| class AbstractEncoder; | ||||
| class AbstractDecoder; | ||||
| class DecoderProto; | ||||
| class EncoderProto; | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createTids990Decoder(const DecoderProto& config); | ||||
| extern std::unique_ptr<AbstractEncoder> createTids990Encoder(const EncoderProto& config); | ||||
|  | ||||
| #endif | ||||
|  | ||||
|  | ||||
							
								
								
									
										26
									
								
								arch/tids990/tids990.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								arch/tids990/tids990.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| import "lib/common.proto"; | ||||
|  | ||||
| message Tids990DecoderProto {} | ||||
| message Tids990EncoderProto { | ||||
| 	optional double track_length_ms = 1 [ default = 166, | ||||
| 		(help) = "length of a track" ]; | ||||
| 	optional int32 sector_count = 2 [ default = 26, | ||||
| 		(help) = "number of sectors per track" ]; | ||||
| 	optional double clock_rate_khz = 3 [ default = 500, | ||||
| 		(help) = "clock rate of data to write" ]; | ||||
| 	optional int32 am1_byte = 4 [ default = 0x2244, | ||||
| 		(help) = "16-bit RAW bit pattern to use for the AM1 ID byte" ]; | ||||
| 	optional int32 am2_byte = 5 [ default = 0x2245, | ||||
| 		(help) = "16-bit RAW bit pattern to use for the AM2 ID byte" ]; | ||||
| 	optional int32 gap1_bytes = 6 [ default = 80, | ||||
| 		(help) = "size of gap 1 (the post-index gap)" ]; | ||||
| 	optional int32 gap2_bytes = 7 [ default = 21, | ||||
| 		(help) = "size of gap 2 (the post-ID gap)" ]; | ||||
| 	optional int32 gap3_bytes = 8 [ default = 51, | ||||
| 		(help) = "size of gap 3 (the post-data or format gap)" ]; | ||||
| 	optional string sector_skew = 9 [ default = "1mhc72nid83oje94pkfa50lgb6", | ||||
| 		(help) = "order to emit sectors" ]; | ||||
| } | ||||
|  | ||||
							
								
								
									
										124
									
								
								arch/victor9k/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										124
									
								
								arch/victor9k/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,124 @@ | ||||
| #include "globals.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "protocol.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "victor9k.h" | ||||
| #include "crc.h" | ||||
| #include "bytes.h" | ||||
| #include "fmt/format.h" | ||||
| #include <string.h> | ||||
| #include <algorithm> | ||||
|  | ||||
| const FluxPattern SECTOR_RECORD_PATTERN(32, VICTOR9K_SECTOR_RECORD); | ||||
| const FluxPattern DATA_RECORD_PATTERN(32, VICTOR9K_DATA_RECORD); | ||||
| const FluxMatchers ANY_RECORD_PATTERN({ &SECTOR_RECORD_PATTERN, &DATA_RECORD_PATTERN }); | ||||
|  | ||||
| static int decode_data_gcr(uint8_t gcr) | ||||
| { | ||||
|     switch (gcr) | ||||
|     { | ||||
| 		#define GCR_ENTRY(gcr, data) \ | ||||
| 			case gcr: return data; | ||||
| 		#include "data_gcr.h" | ||||
| 		#undef GCR_ENTRY | ||||
|     } | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| static Bytes decode(const std::vector<bool>& bits) | ||||
| { | ||||
|     Bytes output; | ||||
|     ByteWriter bw(output); | ||||
|     BitWriter bitw(bw); | ||||
|  | ||||
|     auto ii = bits.begin(); | ||||
|     while (ii != bits.end()) | ||||
|     { | ||||
|         uint8_t inputfifo = 0; | ||||
|         for (size_t i=0; i<5; i++) | ||||
|         { | ||||
|             if (ii == bits.end()) | ||||
|                 break; | ||||
|             inputfifo = (inputfifo<<1) | *ii++; | ||||
|         } | ||||
|  | ||||
|         uint8_t decoded = decode_data_gcr(inputfifo); | ||||
|         bitw.push(decoded, 4); | ||||
|     } | ||||
|     bitw.flush(); | ||||
|  | ||||
|     return output; | ||||
| } | ||||
|  | ||||
| class Victor9kDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	Victor9kDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
|  | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_sector->clock = _fmr->seekToPattern(ANY_RECORD_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_RECORD_PATTERN) | ||||
| 			return SECTOR_RECORD; | ||||
| 		if (matcher == &DATA_RECORD_PATTERN) | ||||
| 			return DATA_RECORD; | ||||
| 		return UNKNOWN_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		/* Skip the sync marker bit. */ | ||||
| 		readRawBits(22); | ||||
|  | ||||
| 		/* Read header. */ | ||||
|  | ||||
| 		auto bytes = decode(readRawBits(4*10)).slice(0, 4); | ||||
|  | ||||
| 		uint8_t rawTrack = bytes[1]; | ||||
| 		_sector->logicalSector = bytes[2]; | ||||
| 		uint8_t gotChecksum = bytes[3]; | ||||
|  | ||||
| 		_sector->logicalTrack = rawTrack & 0x7f; | ||||
| 		_sector->logicalSide = rawTrack >> 7; | ||||
| 		uint8_t wantChecksum = bytes[1] + bytes[2]; | ||||
| 		if ((_sector->logicalSector > 20) || (_sector->logicalTrack > 85) || (_sector->logicalSide > 1)) | ||||
| 			return; | ||||
| 					 | ||||
| 		if (wantChecksum == gotChecksum) | ||||
| 			_sector->status = Sector::DATA_MISSING; /* unintuitive but correct */ | ||||
| 	} | ||||
|  | ||||
|     void decodeDataRecord() | ||||
| 	{ | ||||
| 		/* Skip the sync marker bit. */ | ||||
| 		readRawBits(22); | ||||
|  | ||||
| 		/* Read data. */ | ||||
|  | ||||
| 		auto bytes = decode(readRawBits((VICTOR9K_SECTOR_LENGTH+5)*10)) | ||||
| 			.slice(0, VICTOR9K_SECTOR_LENGTH+5); | ||||
| 		ByteReader br(bytes); | ||||
|  | ||||
| 		/* Check that this is actually a data record. */ | ||||
| 		 | ||||
| 		if (br.read_8() != 8) | ||||
| 			return; | ||||
|  | ||||
| 		_sector->data = br.read(VICTOR9K_SECTOR_LENGTH); | ||||
| 		uint16_t gotChecksum = sumBytes(_sector->data); | ||||
| 		uint16_t wantChecksum = br.read_le16(); | ||||
| 		_sector->status = (gotChecksum == wantChecksum) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createVictor9kDecoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new Victor9kDecoder(config)); | ||||
| } | ||||
|  | ||||
|  | ||||
							
								
								
									
										200
									
								
								arch/victor9k/encoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										200
									
								
								arch/victor9k/encoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,200 @@ | ||||
| #include "globals.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "encoders/encoders.h" | ||||
| #include "victor9k.h" | ||||
| #include "crc.h" | ||||
| #include "sector.h" | ||||
| #include "writer.h" | ||||
| #include "image.h" | ||||
| #include "fmt/format.h" | ||||
| #include "arch/victor9k/victor9k.pb.h" | ||||
| #include "lib/encoders/encoders.pb.h" | ||||
| #include <ctype.h> | ||||
| #include "bytes.h" | ||||
|  | ||||
| static bool lastBit; | ||||
|  | ||||
| static void write_zero_bits(std::vector<bool>& bits, unsigned& cursor, unsigned count) | ||||
| { | ||||
|     while (count--) | ||||
| 	{ | ||||
| 		if (cursor < bits.size()) | ||||
| 			lastBit = bits[cursor++] = 0; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void write_one_bits(std::vector<bool>& bits, unsigned& cursor, unsigned count) | ||||
| { | ||||
|     while (count--) | ||||
| 	{ | ||||
| 		if (cursor < bits.size()) | ||||
| 			lastBit = bits[cursor++] = 1; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, const std::vector<bool>& src) | ||||
| { | ||||
| 	for (bool bit : src) | ||||
| 	{ | ||||
| 		if (cursor < bits.size()) | ||||
| 			lastBit = bits[cursor++] = bit; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, uint64_t data, int width) | ||||
| { | ||||
| 	cursor += width; | ||||
| 	lastBit = data & 1; | ||||
| 	for (int i=0; i<width; i++) | ||||
| 	{ | ||||
| 		unsigned pos = cursor - i - 1; | ||||
| 		if (pos < bits.size()) | ||||
| 			bits[pos] = data & 1; | ||||
| 		data >>= 1; | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static void write_bits(std::vector<bool>& bits, unsigned& cursor, const Bytes& bytes) | ||||
| { | ||||
| 	ByteReader br(bytes); | ||||
| 	BitReader bitr(br); | ||||
|  | ||||
| 	while (!bitr.eof()) | ||||
| 	{ | ||||
| 		if (cursor < bits.size()) | ||||
| 			bits[cursor++] = bitr.get(); | ||||
| 	} | ||||
| } | ||||
|  | ||||
| static int encode_data_gcr(uint8_t data) | ||||
| { | ||||
|     switch (data & 0x0f) | ||||
|     { | ||||
|         #define GCR_ENTRY(gcr, data) \ | ||||
|             case data: return gcr; | ||||
|         #include "data_gcr.h" | ||||
|         #undef GCR_ENTRY | ||||
|     } | ||||
|     return -1; | ||||
| } | ||||
|  | ||||
| static void write_bytes(std::vector<bool>& bits, unsigned& cursor, const Bytes& bytes) | ||||
| { | ||||
|     for (uint8_t b : bytes) | ||||
|     { | ||||
|         write_bits(bits, cursor, encode_data_gcr(b>>4), 5); | ||||
|         write_bits(bits, cursor, encode_data_gcr(b),    5); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static void write_sector(std::vector<bool>& bits, unsigned& cursor, | ||||
| 		const Victor9kEncoderProto::TrackdataProto& trackdata, | ||||
|         const Sector& sector) | ||||
| { | ||||
|     write_one_bits(bits, cursor, trackdata.pre_header_sync_bits()); | ||||
|     write_bits(bits, cursor, VICTOR9K_SECTOR_RECORD, 10); | ||||
|  | ||||
|     uint8_t encodedTrack = sector.logicalTrack | (sector.logicalSide<<7); | ||||
|     uint8_t encodedSector = sector.logicalSector; | ||||
|     write_bytes(bits, cursor, Bytes { | ||||
|         encodedTrack, | ||||
|         encodedSector, | ||||
|         (uint8_t)(encodedTrack + encodedSector), | ||||
|     }); | ||||
|  | ||||
|  | ||||
|     write_zero_bits(bits, cursor, trackdata.post_header_gap_bits()); | ||||
|     write_one_bits(bits, cursor, trackdata.pre_data_sync_bits()); | ||||
|     write_bits(bits, cursor, VICTOR9K_DATA_RECORD, 10); | ||||
|  | ||||
|     write_bytes(bits, cursor, sector.data); | ||||
|  | ||||
|     Bytes checksum(2); | ||||
|     checksum.writer().write_le16(sumBytes(sector.data)); | ||||
|     write_bytes(bits, cursor, checksum); | ||||
|  | ||||
|     write_zero_bits(bits, cursor, trackdata.post_data_gap_bits()); | ||||
| } | ||||
|  | ||||
| class Victor9kEncoder : public AbstractEncoder | ||||
| { | ||||
| public: | ||||
| 	Victor9kEncoder(const EncoderProto& config): | ||||
|         AbstractEncoder(config), | ||||
| 		_config(config.victor9k()) | ||||
| 	{} | ||||
|  | ||||
| private: | ||||
|  | ||||
| 	void getTrackFormat(Victor9kEncoderProto::TrackdataProto& trackdata, unsigned cylinder, unsigned head) | ||||
| 	{ | ||||
| 		trackdata.Clear(); | ||||
| 		for (const auto& f : _config.trackdata()) | ||||
| 		{ | ||||
| 			if (f.has_min_cylinder() && (cylinder < f.min_cylinder())) | ||||
| 				continue; | ||||
| 			if (f.has_max_cylinder() && (cylinder > f.max_cylinder())) | ||||
| 				continue; | ||||
| 			if (f.has_head() && (head != f.head())) | ||||
| 				continue; | ||||
|  | ||||
| 			trackdata.MergeFrom(f); | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| public: | ||||
| 	std::vector<std::shared_ptr<Sector>> collectSectors(int physicalTrack, int physicalSide, const Image& image) override | ||||
| 	{ | ||||
| 		std::vector<std::shared_ptr<Sector>> sectors; | ||||
|  | ||||
| 		Victor9kEncoderProto::TrackdataProto trackdata; | ||||
| 		getTrackFormat(trackdata, physicalTrack, physicalSide); | ||||
|  | ||||
|         for (int i = 0; i < trackdata.sector_range().sector_count(); i++) | ||||
|         { | ||||
|             int sectorId = trackdata.sector_range().start_sector() + i; | ||||
| 			const auto& sector = image.get(physicalTrack, physicalSide, sectorId); | ||||
| 			if (sector) | ||||
| 				sectors.push_back(sector); | ||||
|         } | ||||
|  | ||||
| 		return sectors; | ||||
| 	} | ||||
|  | ||||
|     std::unique_ptr<Fluxmap> encode(int physicalTrack, int physicalSide, | ||||
|             const std::vector<std::shared_ptr<Sector>>& sectors, const Image& image) override | ||||
|     { | ||||
| 		Victor9kEncoderProto::TrackdataProto trackdata; | ||||
| 		getTrackFormat(trackdata, physicalTrack, physicalSide); | ||||
|  | ||||
|         unsigned bitsPerRevolution = trackdata.original_data_rate_khz() * trackdata.original_period_ms(); | ||||
|         std::vector<bool> bits(bitsPerRevolution); | ||||
|         double clockRateUs = 166666.0 / bitsPerRevolution; | ||||
|         unsigned cursor = 0; | ||||
|  | ||||
|         fillBitmapTo(bits, cursor, trackdata.post_index_gap_us() / clockRateUs, { true, false }); | ||||
|         lastBit = false; | ||||
|  | ||||
|         for (const auto& sector : sectors) | ||||
|             write_sector(bits, cursor, trackdata, *sector); | ||||
|  | ||||
|         if (cursor >= bits.size()) | ||||
|             Error() << fmt::format("track data overrun by {} bits", cursor - bits.size()); | ||||
|         fillBitmapTo(bits, cursor, bits.size(), { true, false }); | ||||
|  | ||||
|         std::unique_ptr<Fluxmap> fluxmap(new Fluxmap); | ||||
|         fluxmap->appendBits(bits, clockRateUs*1e3); | ||||
|         return fluxmap; | ||||
|     } | ||||
|  | ||||
| private: | ||||
| 	const Victor9kEncoderProto& _config; | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractEncoder> createVictor9kEncoder(const EncoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractEncoder>(new Victor9kEncoder(config)); | ||||
| } | ||||
|  | ||||
| // vim: sw=4 ts=4 et | ||||
|  | ||||
							
								
								
									
										22
									
								
								arch/victor9k/victor9k.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								arch/victor9k/victor9k.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | ||||
| #ifndef VICTOR9K_H | ||||
| #define VICTOR9K_H | ||||
|  | ||||
| class AbstractEncoder; | ||||
| class AbstractDecoder; | ||||
| class EncoderProto; | ||||
| class DecoderProto; | ||||
|  | ||||
| /* ... 1101 0101 0111 | ||||
|  *       ^^ ^^^^ ^^^^ ten bit IO byte */ | ||||
| #define VICTOR9K_SECTOR_RECORD 0xfffffd57  | ||||
|  | ||||
| /* ... 1101 0100 1001 | ||||
|  *       ^^ ^^^^ ^^^^ ten bit IO byte */ | ||||
| #define VICTOR9K_DATA_RECORD   0xfffffd49 | ||||
|  | ||||
| #define VICTOR9K_SECTOR_LENGTH 512 | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createVictor9kDecoder(const DecoderProto& config); | ||||
| extern std::unique_ptr<AbstractEncoder> createVictor9kEncoder(const EncoderProto& config); | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										32
									
								
								arch/victor9k/victor9k.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								arch/victor9k/victor9k.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,32 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| import "lib/common.proto"; | ||||
|  | ||||
| message Victor9kDecoderProto {} | ||||
|  | ||||
| // NEXT: 12 | ||||
| message Victor9kEncoderProto { | ||||
| 	message TrackdataProto { | ||||
| 		message SectorRangeProto { | ||||
| 			optional int32 start_sector = 1 [(help) = "first sector ID on track"]; | ||||
| 			optional int32 sector_count = 2 [(help) = "number of sectors on track"]; | ||||
| 		} | ||||
|  | ||||
| 		optional int32 min_cylinder = 1            [(help) = "minimum cylinder this format applies to"]; | ||||
| 		optional int32 max_cylinder = 2            [(help) = "maximum cylinder this format applies to"]; | ||||
| 		optional int32 head = 3                    [(help) = "which head this format applies to"]; | ||||
|  | ||||
| 		optional double original_period_ms = 4     [(help) = "original rotational period of this cylinder"]; | ||||
| 		optional double original_data_rate_khz = 5 [(help) = "original data rate of this cylinder"]; | ||||
| 		optional double post_index_gap_us = 6      [(help) = "size of post-index gap"]; | ||||
| 		optional int32 pre_header_sync_bits = 10   [(help) = "number of sync bits before the sector header"]; | ||||
| 		optional int32 pre_data_sync_bits = 8      [(help) = "number of sync bits before the sector data"]; | ||||
| 		optional int32 post_data_gap_bits = 9      [(help) = "size of gap between data and the next header"]; | ||||
| 		optional int32 post_header_gap_bits = 11   [(help) = "size of gap between header and the data"]; | ||||
|  | ||||
| 		optional SectorRangeProto sector_range = 7 [(help) = "write these sectors on each track"]; | ||||
| 	} | ||||
|  | ||||
| 	repeated TrackdataProto trackdata = 1; | ||||
| } | ||||
|  | ||||
							
								
								
									
										61
									
								
								arch/zilogmcz/decoder.cc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								arch/zilogmcz/decoder.cc
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | ||||
| #include "globals.h" | ||||
| #include "fluxmap.h" | ||||
| #include "decoders/fluxmapreader.h" | ||||
| #include "protocol.h" | ||||
| #include "decoders/decoders.h" | ||||
| #include "sector.h" | ||||
| #include "zilogmcz.h" | ||||
| #include "bytes.h" | ||||
| #include "crc.h" | ||||
| #include "fmt/format.h" | ||||
| #include <string.h> | ||||
| #include <algorithm> | ||||
|  | ||||
| static const FluxPattern SECTOR_START_PATTERN(16, 0xaaab); | ||||
|  | ||||
| class ZilogMczDecoder : public AbstractDecoder | ||||
| { | ||||
| public: | ||||
| 	ZilogMczDecoder(const DecoderProto& config): | ||||
| 		AbstractDecoder(config) | ||||
| 	{} | ||||
|  | ||||
|     RecordType advanceToNextRecord() | ||||
| 	{ | ||||
| 		const FluxMatcher* matcher = nullptr; | ||||
| 		_fmr->seekToIndexMark(); | ||||
| 		_sector->clock = _fmr->seekToPattern(SECTOR_START_PATTERN, matcher); | ||||
| 		if (matcher == &SECTOR_START_PATTERN) | ||||
| 			return SECTOR_RECORD; | ||||
| 		return UNKNOWN_RECORD; | ||||
| 	} | ||||
|  | ||||
|     void decodeSectorRecord() | ||||
| 	{ | ||||
| 		readRawBits(14); | ||||
|  | ||||
| 		auto rawbits = readRawBits(140*16); | ||||
| 		auto bytes = decodeFmMfm(rawbits).slice(0, 140); | ||||
| 		ByteReader br(bytes); | ||||
|  | ||||
| 		_sector->logicalSector = br.read_8() & 0x1f; | ||||
| 		_sector->logicalSide = 0; | ||||
| 		_sector->logicalTrack = br.read_8() & 0x7f; | ||||
| 		if (_sector->logicalSector > 31) | ||||
| 			return; | ||||
| 		if (_sector->logicalTrack > 80) | ||||
| 			return; | ||||
|  | ||||
| 		_sector->data = br.read(132); | ||||
| 		uint16_t wantChecksum = br.read_be16(); | ||||
| 		uint16_t gotChecksum = crc16(MODBUS_POLY, 0x0000, bytes.slice(0, 134)); | ||||
|  | ||||
| 		_sector->status = (wantChecksum == gotChecksum) ? Sector::OK : Sector::BAD_CHECKSUM; | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| std::unique_ptr<AbstractDecoder> createZilogMczDecoder(const DecoderProto& config) | ||||
| { | ||||
| 	return std::unique_ptr<AbstractDecoder>(new ZilogMczDecoder(config)); | ||||
| } | ||||
|  | ||||
							
								
								
									
										8
									
								
								arch/zilogmcz/zilogmcz.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								arch/zilogmcz/zilogmcz.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,8 @@ | ||||
| #ifndef ZILOGMCZ_H | ||||
| #define ZILOGMCZ_H | ||||
|  | ||||
| extern std::unique_ptr<AbstractDecoder> createZilogMczDecoder(const DecoderProto& config); | ||||
|  | ||||
| #endif | ||||
|  | ||||
|  | ||||
							
								
								
									
										4
									
								
								arch/zilogmcz/zilogmcz.proto
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								arch/zilogmcz/zilogmcz.proto
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | ||||
| syntax = "proto2"; | ||||
|  | ||||
| message ZilogMczDecoderProto {} | ||||
|  | ||||
							
								
								
									
										2
									
								
								dep/agg/AUTHORS
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/agg/AUTHORS
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,2 @@ | ||||
| Anti-Grain Geometry - Version 2.4 | ||||
| Copyright (C) 2002-2005 Maxim Shemanarev (McSeem)  | ||||
							
								
								
									
										63
									
								
								dep/agg/README
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								dep/agg/README
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,63 @@ | ||||
| The Anti-Grain Geometry Project | ||||
| A high quality rendering engine for C++ | ||||
| http://antigrain.com | ||||
|  | ||||
| Anti-Grain Geometry - Version 2.4 | ||||
| Copyright (C) 2002-2005 Maxim Shemanarev (McSeem) | ||||
|  | ||||
| Permission to copy, use, modify, sell and distribute this software | ||||
| is granted provided this copyright notice appears in all copies. | ||||
| This software is provided "as is" without express or implied | ||||
| warranty, and with no claim as to its suitability for any purpose. | ||||
|  | ||||
| --------------------------------- | ||||
|  | ||||
| Use automake to build the library. | ||||
|  | ||||
| If automake is not available you still can use the old make. | ||||
| There is a very simple Makefile that can be used. Note that | ||||
| if you use automake it will overwrite Makefile. | ||||
|  | ||||
| --------------------------------- | ||||
|  | ||||
| If building on AmigaOS 4.0 or higher type the following for | ||||
| instructions on what targets are available. | ||||
|  make -f Makefile.AmigaOS | ||||
|  | ||||
| To just build and install AGG into the standard AmigaOS SDK | ||||
| ready for use type: | ||||
|  make -f Makefile.AmigaOS install | ||||
|  | ||||
| If you just want to build one demo (e.g. lion) use: | ||||
|  make -f Makefile.AmigaOS bin/lion | ||||
|  | ||||
| If you have any questions about the AmigaOS port please | ||||
| contact Steven Solie (ssolie@telus.net) for help. | ||||
|  | ||||
| --------------------------------- | ||||
|  | ||||
| To build all examples using SDL (Mac or Linux) just type: | ||||
|  | ||||
| cd /examples/sdl | ||||
| make | ||||
|  | ||||
| Individual examples can be built with | ||||
|  | ||||
| make aa_test | ||||
|  | ||||
| In the same way the native Carbon examples can be built with | ||||
|  | ||||
| cd /examples/macosx_carbon | ||||
| make | ||||
|  | ||||
| In both cases the static library will be built (if it was not already)  | ||||
| from the existing global Makefile in /src/. | ||||
|  | ||||
| The Makefiles for both SDL and Carbon will also attempt to download the  | ||||
| required .bmp files if they are not found in the system for a given  | ||||
| example. If the files could not be fetched (wget) the user will receive  | ||||
| a message explaining where to download the samples from (sphere.bmp,  | ||||
| etc.)  Since all programs reside in the same directory there is no need  | ||||
| to duplicate the .bmp files for each program that needs to use them. | ||||
|  | ||||
| --------------------------------- | ||||
							
								
								
									
										7
									
								
								dep/agg/README.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								dep/agg/README.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| This is a vary stripped down copy of the Anti-Grain Antialiasing graphics | ||||
| rendering library --- I've removed all the platform-specific and control stuff | ||||
| so that it can be used to generate memory images only. | ||||
|  | ||||
| The original AGG site is dead, so this version is cloned from | ||||
| https://github.com/NNemec/antigrain. | ||||
|  | ||||
							
								
								
									
										7
									
								
								dep/agg/UPSTREAM.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								dep/agg/UPSTREAM.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | ||||
| This is a vary stripped down copy of the Anti-Grain Antialiasing graphics | ||||
| rendering library --- I've removed all the platform-specific and control stuff | ||||
| so that it can be used to generate memory images only. | ||||
|  | ||||
| The original AGG site is dead, so this version is cloned from | ||||
| https://github.com/NNemec/antigrain. | ||||
|  | ||||
							
								
								
									
										49
									
								
								dep/agg/include/Makefile.am
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										49
									
								
								dep/agg/include/Makefile.am
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,49 @@ | ||||
| SUBDIRS = ctrl util platform  | ||||
|  | ||||
| aggincludedir = $(includedir)/agg2 | ||||
| agginclude_HEADERS = \ | ||||
| 	agg_alpha_mask_u8.h          agg_glyph_raster_bin.h          agg_span_allocator.h \ | ||||
| 	agg_arc.h                    agg_gsv_text.h                  agg_span_converter.h \ | ||||
| 	agg_array.h                  agg_image_accessors.h           agg_span_gouraud.h \ | ||||
| 	agg_arrowhead.h              agg_image_filters.h             agg_span_gouraud_gray.h \ | ||||
| 	agg_basics.h                 agg_line_aa_basics.h            agg_span_gouraud_rgba.h \ | ||||
| 	agg_bezier_arc.h             agg_math.h                      agg_span_gradient.h \ | ||||
| 	agg_bitset_iterator.h        agg_blur.h                      agg_math_stroke.h \ | ||||
| 	agg_span_gradient_alpha.h    agg_gradient_lut.h \ | ||||
| 	agg_bounding_rect.h          agg_path_length.h               agg_span_image_filter.h \ | ||||
| 	agg_bspline.h                agg_path_storage.h              agg_span_image_filter_gray.h \ | ||||
| 	agg_clip_liang_barsky.h      agg_path_storage_integer.h      agg_span_image_filter_rgb.h \ | ||||
| 	agg_color_gray.h             agg_pattern_filters_rgba.h      agg_span_image_filter_rgba.h \ | ||||
| 	agg_color_rgba.h             agg_pixfmt_amask_adaptor.h      agg_span_interpolator_adaptor.h \ | ||||
| 	agg_config.h                 agg_pixfmt_gray.h               agg_span_interpolator_linear.h \ | ||||
| 	agg_conv_adaptor_vcgen.h     agg_pixfmt_rgb.h                agg_span_interpolator_persp.h \ | ||||
| 	agg_conv_adaptor_vpgen.h     agg_pixfmt_rgb_packed.h         agg_span_interpolator_trans.h \ | ||||
| 	agg_conv_bspline.h           agg_pixfmt_rgba.h               agg_pixfmt_transposer.h \ | ||||
| 	agg_span_pattern_gray.h \ | ||||
| 	agg_conv_clip_polygon.h      agg_rasterizer_cells_aa.h       agg_span_pattern_rgb.h \ | ||||
| 	agg_conv_clip_polyline.h     agg_rasterizer_compound_aa.h    agg_span_pattern_rgba.h \ | ||||
| 	agg_conv_close_polygon.h     agg_rasterizer_outline.h        agg_span_solid.h \ | ||||
| 	agg_conv_concat.h            agg_rasterizer_outline_aa.h     agg_span_subdiv_adaptor.h \ | ||||
| 	agg_conv_contour.h           agg_rasterizer_scanline_aa.h    agg_trans_affine.h \ | ||||
| 	agg_conv_curve.h             agg_rasterizer_sl_clip.h        agg_trans_bilinear.h \ | ||||
| 	agg_conv_dash.h              agg_renderer_base.h             agg_trans_double_path.h \ | ||||
| 	agg_conv_gpc.h               agg_renderer_markers.h \ | ||||
| 	agg_conv_marker.h            agg_renderer_mclip.h            agg_trans_perspective.h \ | ||||
| 	agg_conv_marker_adaptor.h    agg_renderer_outline_aa.h       agg_trans_single_path.h \ | ||||
| 	agg_conv_segmentator.h       agg_renderer_outline_image.h    agg_trans_viewport.h \ | ||||
| 	agg_conv_shorten_path.h      agg_renderer_primitives.h       agg_trans_warp_magnifier.h \ | ||||
| 	agg_conv_smooth_poly1.h      agg_renderer_raster_text.h      agg_vcgen_bspline.h \ | ||||
| 	agg_conv_stroke.h            agg_renderer_scanline.h         agg_vcgen_contour.h \ | ||||
| 	agg_conv_transform.h         agg_rendering_buffer.h          agg_vcgen_dash.h \ | ||||
| 	agg_conv_unclose_polygon.h   agg_rendering_buffer_dynarow.h  agg_vcgen_markers_term.h \ | ||||
| 	agg_curves.h                 agg_rounded_rect.h              agg_vcgen_smooth_poly1.h \ | ||||
| 	                             agg_scanline_bin.h              agg_vcgen_stroke.h \ | ||||
| 	agg_dda_line.h               agg_scanline_boolean_algebra.h  agg_vcgen_vertex_sequence.h \ | ||||
| 	agg_ellipse.h                agg_scanline_p.h                agg_vertex_sequence.h \ | ||||
| 	agg_ellipse_bresenham.h      agg_scanline_storage_aa.h       agg_vpgen_clip_polygon.h \ | ||||
| 	agg_embedded_raster_fonts.h  agg_scanline_storage_bin.h      agg_vpgen_clip_polyline.h \ | ||||
| 	agg_font_cache_manager.h     agg_scanline_u.h                agg_vpgen_segmentator.h \ | ||||
| 	agg_gamma_functions.h        agg_shorten_path.h \ | ||||
| 	agg_gamma_lut.h              agg_simul_eq.h \ | ||||
| 	agg_font_cache_manager2.h    agg_pixfmt_base.h               agg_rasterizer_scanline_aa_nogamma.h \ | ||||
| 	agg_span_gradient_contour.h  agg_span_gradient_image.h | ||||
							
								
								
									
										568
									
								
								dep/agg/include/agg2d.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										568
									
								
								dep/agg/include/agg2d.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,568 @@ | ||||
| //---------------------------------------------------------------------------- | ||||
| // Agg2D - Version 1.0 | ||||
| // Based on Anti-Grain Geometry | ||||
| // Copyright (C) 2005 Maxim Shemanarev (http://www.antigrain.com) | ||||
| // | ||||
| // Permission to copy, use, modify, sell and distribute this software | ||||
| // is granted provided this copyright notice appears in all copies. | ||||
| // This software is provided "as is" without express or implied | ||||
| // warranty, and with no claim as to its suitability for any purpose. | ||||
| // | ||||
| //---------------------------------------------------------------------------- | ||||
| // Contact: mcseem@antigrain.com | ||||
| //          mcseemagg@yahoo.com | ||||
| //          http://www.antigrain.com | ||||
| //---------------------------------------------------------------------------- | ||||
| //---------------------------------------------------------------------------- | ||||
| // | ||||
| //	25 Jan 2007 - Ported to AGG 2.4 Jerry Evans (jerry@novadsp.com) | ||||
| // | ||||
| //---------------------------------------------------------------------------- | ||||
|  | ||||
| #ifndef AGG2D_INCLUDED | ||||
| #define AGG2D_INCLUDED | ||||
|  | ||||
| // With this define uncommented you can use floating-point pixel format | ||||
| //#define AGG2D_USE_FLOAT_FORMAT | ||||
|  | ||||
| #include "agg_basics.h" | ||||
| #include "agg_trans_affine.h" | ||||
| #include "agg_trans_viewport.h" | ||||
| #include "agg_path_storage.h" | ||||
| #include "agg_conv_stroke.h" | ||||
| #include "agg_conv_transform.h" | ||||
| #include "agg_conv_curve.h" | ||||
| #include "agg_rendering_buffer.h" | ||||
| #include "agg_renderer_base.h" | ||||
| #include "agg_renderer_scanline.h" | ||||
| #include "agg_span_gradient.h" | ||||
| #include "agg_span_image_filter_rgba.h" | ||||
| #include "agg_span_allocator.h" | ||||
| #include "agg_span_converter.h" | ||||
| #include "agg_span_interpolator_linear.h" | ||||
| #include "agg_rasterizer_scanline_aa.h" | ||||
| #include "agg_gamma_functions.h" | ||||
| #include "agg_scanline_u.h" | ||||
| #include "agg_bezier_arc.h" | ||||
| #include "agg_rounded_rect.h" | ||||
| #include "agg_font_cache_manager.h" | ||||
|  | ||||
| #include "agg_pixfmt_rgba.h" | ||||
| #include "agg_image_accessors.h" | ||||
| #include <string> | ||||
|  | ||||
| class Agg2D | ||||
| { | ||||
| #ifdef AGG2D_USE_FLOAT_FORMAT | ||||
|     typedef agg::rgba32 ColorType; | ||||
| #else | ||||
|     typedef agg::rgba8  ColorType; | ||||
| #endif | ||||
|     typedef agg::order_bgra ComponentOrder; // Platform dependent! | ||||
|     typedef agg::blender_rgba<ColorType, ComponentOrder>             Blender; | ||||
|     typedef agg::comp_op_adaptor_rgba<ColorType, ComponentOrder>     BlenderComp; | ||||
|     typedef agg::blender_rgba_pre<ColorType, ComponentOrder>         BlenderPre; | ||||
|     typedef agg::comp_op_adaptor_rgba_pre<ColorType, ComponentOrder> BlenderCompPre; | ||||
|  | ||||
|     typedef agg::pixfmt_alpha_blend_rgba<Blender, agg::rendering_buffer>         PixFormat; | ||||
|     typedef agg::pixfmt_custom_blend_rgba<BlenderComp, agg::rendering_buffer>    PixFormatComp; | ||||
|     typedef agg::pixfmt_alpha_blend_rgba<BlenderPre, agg::rendering_buffer>      PixFormatPre; | ||||
|     typedef agg::pixfmt_custom_blend_rgba<BlenderCompPre, agg::rendering_buffer> PixFormatCompPre; | ||||
|  | ||||
|     typedef agg::renderer_base<PixFormat>        RendererBase; | ||||
|     typedef agg::renderer_base<PixFormatComp>    RendererBaseComp; | ||||
|     typedef agg::renderer_base<PixFormatPre>     RendererBasePre; | ||||
|     typedef agg::renderer_base<PixFormatCompPre> RendererBaseCompPre; | ||||
|  | ||||
|     typedef agg::renderer_scanline_aa_solid<RendererBase>     RendererSolid; | ||||
|     typedef agg::renderer_scanline_aa_solid<RendererBaseComp> RendererSolidComp; | ||||
|  | ||||
|     typedef agg::span_allocator<ColorType> SpanAllocator; | ||||
|     typedef agg::pod_auto_array<ColorType, 256> GradientArray; | ||||
|  | ||||
|     typedef agg::span_gradient<ColorType, agg::span_interpolator_linear<>, agg::gradient_x,      GradientArray> LinearGradientSpan; | ||||
|     typedef agg::span_gradient<ColorType, agg::span_interpolator_linear<>, agg::gradient_circle, GradientArray> RadialGradientSpan; | ||||
|  | ||||
|     typedef agg::conv_curve<agg::path_storage>    ConvCurve; | ||||
|     typedef agg::conv_stroke<ConvCurve>           ConvStroke; | ||||
|     typedef agg::conv_transform<ConvCurve>        PathTransform; | ||||
|     typedef agg::conv_transform<ConvStroke>       StrokeTransform; | ||||
|  | ||||
|     enum Gradient | ||||
|     { | ||||
|         Solid, | ||||
|         Linear, | ||||
|         Radial | ||||
|     }; | ||||
|  | ||||
| public: | ||||
|     friend class Agg2DRenderer; | ||||
|  | ||||
|     // Use srgba8 as the "user" color type, even though the underlying color type  | ||||
|     // might be something else, such as rgba32. This allows code based on  | ||||
|     // 8-bit sRGB values to carry on working as before. | ||||
|     typedef agg::srgba8       Color; | ||||
|     typedef agg::rect_i       Rect; | ||||
|     typedef agg::rect_d       RectD; | ||||
|     typedef agg::trans_affine Affine; | ||||
|  | ||||
|     enum LineJoin | ||||
|     { | ||||
|         JoinMiter = agg::miter_join, | ||||
|         JoinRound = agg::round_join, | ||||
|         JoinBevel = agg::bevel_join | ||||
|     }; | ||||
|  | ||||
|     enum LineCap | ||||
|     { | ||||
|         CapButt   = agg::butt_cap, | ||||
|         CapSquare = agg::square_cap, | ||||
|         CapRound  = agg::round_cap | ||||
|     }; | ||||
|  | ||||
|     enum TextAlignment | ||||
|     { | ||||
|         AlignLeft, | ||||
|         AlignRight, | ||||
|         AlignCenter, | ||||
|     }; | ||||
|  | ||||
|  | ||||
|     enum DrawPathFlag | ||||
|     { | ||||
|         FillOnly, | ||||
|         StrokeOnly, | ||||
|         FillAndStroke, | ||||
|         FillWithLineColor | ||||
|     }; | ||||
|  | ||||
|     enum ViewportOption | ||||
|     { | ||||
|         Anisotropic, | ||||
|         XMinYMin, | ||||
|         XMidYMin, | ||||
|         XMaxYMin, | ||||
|         XMinYMid, | ||||
|         XMidYMid, | ||||
|         XMaxYMid, | ||||
|         XMinYMax, | ||||
|         XMidYMax, | ||||
|         XMaxYMax | ||||
|     }; | ||||
|  | ||||
|     struct Transformations | ||||
|     { | ||||
|         double affineMatrix[6]; | ||||
|     }; | ||||
|  | ||||
|  | ||||
|     struct Image | ||||
|     { | ||||
|         agg::rendering_buffer renBuf; | ||||
|  | ||||
|         Image() {} | ||||
|         Image(unsigned char* buf, unsigned width, unsigned height, int stride) : | ||||
|             renBuf(buf, width, height, stride) {} | ||||
|         void attach(unsigned char* buf, unsigned width, unsigned height, int stride) | ||||
|         { | ||||
|             renBuf.attach(buf, width, height, stride); | ||||
|         } | ||||
|         int width()  const { return renBuf.width(); } | ||||
|         int height() const { return renBuf.height(); } | ||||
|         void premultiply(); | ||||
|         void demultiply(); | ||||
|     }; | ||||
|  | ||||
|     enum ImageFilter | ||||
|     { | ||||
|         NoFilter, | ||||
|         Bilinear, | ||||
|         Hanning, | ||||
|         Hermite, | ||||
|         Quadric, | ||||
|         Bicubic, | ||||
|         Catrom, | ||||
|         Spline16, | ||||
|         Spline36, | ||||
|         Blackman144 | ||||
|     }; | ||||
|  | ||||
|     enum ImageResample | ||||
|     { | ||||
|         NoResample, | ||||
|         ResampleAlways, | ||||
|         ResampleOnZoomOut | ||||
|     }; | ||||
|  | ||||
|     enum FontCacheType | ||||
|     { | ||||
|         RasterFontCache, | ||||
|         VectorFontCache | ||||
|     }; | ||||
|  | ||||
|     enum BlendMode | ||||
|     { | ||||
|         BlendAlpha      = agg::end_of_comp_op_e, | ||||
|         BlendClear      = agg::comp_op_clear, | ||||
|         BlendSrc        = agg::comp_op_src, | ||||
|         BlendDst        = agg::comp_op_dst, | ||||
|         BlendSrcOver    = agg::comp_op_src_over, | ||||
|         BlendDstOver    = agg::comp_op_dst_over, | ||||
|         BlendSrcIn      = agg::comp_op_src_in, | ||||
|         BlendDstIn      = agg::comp_op_dst_in, | ||||
|         BlendSrcOut     = agg::comp_op_src_out, | ||||
|         BlendDstOut     = agg::comp_op_dst_out, | ||||
|         BlendSrcAtop    = agg::comp_op_src_atop, | ||||
|         BlendDstAtop    = agg::comp_op_dst_atop, | ||||
|         BlendXor        = agg::comp_op_xor, | ||||
|         BlendAdd        = agg::comp_op_plus, | ||||
|         BlendMultiply   = agg::comp_op_multiply, | ||||
|         BlendScreen     = agg::comp_op_screen, | ||||
|         BlendOverlay    = agg::comp_op_overlay, | ||||
|         BlendDarken     = agg::comp_op_darken, | ||||
|         BlendLighten    = agg::comp_op_lighten, | ||||
|         BlendColorDodge = agg::comp_op_color_dodge, | ||||
|         BlendColorBurn  = agg::comp_op_color_burn, | ||||
|         BlendHardLight  = agg::comp_op_hard_light, | ||||
|         BlendSoftLight  = agg::comp_op_soft_light, | ||||
|         BlendDifference = agg::comp_op_difference, | ||||
|         BlendExclusion  = agg::comp_op_exclusion, | ||||
|     }; | ||||
|  | ||||
|     enum Direction | ||||
|     { | ||||
|         CW, CCW | ||||
|     }; | ||||
|  | ||||
|     ~Agg2D(); | ||||
|     Agg2D(); | ||||
|  | ||||
|     // Setup | ||||
|     //----------------------- | ||||
|     void  attach(unsigned char* buf, unsigned width, unsigned height, int stride); | ||||
|     void  attach(Image& img); | ||||
|  | ||||
|     void  clipBox(double x1, double y1, double x2, double y2); | ||||
|     RectD clipBox() const; | ||||
|  | ||||
|     void  clearAll(Color c); | ||||
|     void  clearAll(unsigned r, unsigned g, unsigned b, unsigned a = 255); | ||||
|  | ||||
|     void  clearClipBox(Color c); | ||||
|     void  clearClipBox(unsigned r, unsigned g, unsigned b, unsigned a = 255); | ||||
|  | ||||
|     // Conversions | ||||
|     //----------------------- | ||||
|     void   worldToScreen(double& x, double& y) const; | ||||
|     void   screenToWorld(double& x, double& y) const; | ||||
|     double worldToScreen(double scalar) const; | ||||
|     double screenToWorld(double scalar) const; | ||||
|     void   alignPoint(double& x, double& y) const; | ||||
|     bool   inBox(double worldX, double worldY) const; | ||||
|  | ||||
|     // General Attributes | ||||
|     //----------------------- | ||||
|     void blendMode(BlendMode m); | ||||
|     BlendMode blendMode() const; | ||||
|  | ||||
|     void imageBlendMode(BlendMode m); | ||||
|     BlendMode imageBlendMode() const; | ||||
|  | ||||
|     void imageBlendColor(Color c); | ||||
|     void imageBlendColor(unsigned r, unsigned g, unsigned b, unsigned a = 255); | ||||
|     Color imageBlendColor() const; | ||||
|  | ||||
|     void masterAlpha(double a); | ||||
|     double masterAlpha() const; | ||||
|  | ||||
|     void antiAliasGamma(double g); | ||||
|     double antiAliasGamma() const; | ||||
|  | ||||
|     void fillColor(Color c); | ||||
|     void fillColor(unsigned r, unsigned g, unsigned b, unsigned a = 255); | ||||
|     void noFill(); | ||||
|  | ||||
|     void lineColor(Color c); | ||||
|     void lineColor(unsigned r, unsigned g, unsigned b, unsigned a = 255); | ||||
|     void noLine(); | ||||
|  | ||||
|     Color fillColor() const; | ||||
|     Color lineColor() const; | ||||
|  | ||||
|     void fillLinearGradient(double x1, double y1, double x2, double y2, Color c1, Color c2, double profile=1.0); | ||||
|     void lineLinearGradient(double x1, double y1, double x2, double y2, Color c1, Color c2, double profile=1.0); | ||||
|  | ||||
|     void fillRadialGradient(double x, double y, double r, Color c1, Color c2, double profile=1.0); | ||||
|     void lineRadialGradient(double x, double y, double r, Color c1, Color c2, double profile=1.0); | ||||
|  | ||||
|     void fillRadialGradient(double x, double y, double r, Color c1, Color c2, Color c3); | ||||
|     void lineRadialGradient(double x, double y, double r, Color c1, Color c2, Color c3); | ||||
|  | ||||
|     void fillRadialGradient(double x, double y, double r); | ||||
|     void lineRadialGradient(double x, double y, double r); | ||||
|  | ||||
|     void lineWidth(double w); | ||||
|     double lineWidth(double w) const; | ||||
|  | ||||
|     void lineCap(LineCap cap); | ||||
|     LineCap lineCap() const; | ||||
|  | ||||
|     void lineJoin(LineJoin join); | ||||
|     LineJoin lineJoin() const; | ||||
|  | ||||
|     void fillEvenOdd(bool evenOddFlag); | ||||
|     bool fillEvenOdd() const; | ||||
|  | ||||
| 	void textAlignment(TextAlignment alignment); | ||||
| 	void textSize(double sizeX, double sizeY); | ||||
| 	inline void textSize(double size) { textSize(size, size); } | ||||
|  | ||||
|     // Transformations | ||||
|     //----------------------- | ||||
|     Transformations transformations() const; | ||||
|     void transformations(const Transformations& tr); | ||||
|     void resetTransformations(); | ||||
|     void affine(const Affine& tr); | ||||
|     void affine(const Transformations& tr); | ||||
|     void rotate(double angle); | ||||
|     void scale(double sx, double sy); | ||||
|     void skew(double sx, double sy); | ||||
|     void translate(double x, double y); | ||||
|     void parallelogram(double x1, double y1, double x2, double y2, const double* para); | ||||
|     void viewport(double worldX1,  double worldY1,  double worldX2,  double worldY2, | ||||
|                   double screenX1, double screenY1, double screenX2, double screenY2, | ||||
|                   ViewportOption opt=XMidYMid); | ||||
|  | ||||
|     // Basic Shapes | ||||
|     //----------------------- | ||||
|     void line(double x1, double y1, double x2, double y2); | ||||
|     void triangle(double x1, double y1, double x2, double y2, double x3, double y3); | ||||
|     void rectangle(double x1, double y1, double x2, double y2); | ||||
|     void roundedRect(double x1, double y1, double x2, double y2, double r); | ||||
|     void roundedRect(double x1, double y1, double x2, double y2, double rx, double ry); | ||||
|     void roundedRect(double x1, double y1, double x2, double y2, | ||||
|                      double rxBottom, double ryBottom, | ||||
|                      double rxTop,    double ryTop); | ||||
|     void ellipse(double cx, double cy, double rx, double ry); | ||||
|     void arc(double cx, double cy, double rx, double ry, double start, double sweep); | ||||
|     void star(double cx, double cy, double r1, double r2, double startAngle, int numRays); | ||||
|     void curve(double x1, double y1, double x2, double y2, double x3, double y3); | ||||
|     void curve(double x1, double y1, double x2, double y2, double x3, double y3, double x4, double y4); | ||||
|     void polygon(double* xy, int numPoints); | ||||
|     void polyline(double* xy, int numPoints); | ||||
|  | ||||
|     // Path commands | ||||
|     //----------------------- | ||||
|     void resetPath(); | ||||
|  | ||||
|     void moveTo(double x, double y); | ||||
|     void moveRel(double dx, double dy); | ||||
|  | ||||
|     void lineTo(double x, double y); | ||||
|     void lineRel(double dx, double dy); | ||||
|  | ||||
|     void horLineTo(double x); | ||||
|     void horLineRel(double dx); | ||||
|  | ||||
|     void verLineTo(double y); | ||||
|     void verLineRel(double dy); | ||||
|  | ||||
|     void arcTo(double rx, double ry, | ||||
|                double angle, | ||||
|                bool largeArcFlag, | ||||
|                bool sweepFlag, | ||||
|                double x, double y); | ||||
|  | ||||
|     void arcRel(double rx, double ry, | ||||
|                 double angle, | ||||
|                 bool largeArcFlag, | ||||
|                 bool sweepFlag, | ||||
|                 double dx, double dy); | ||||
|  | ||||
|     void quadricCurveTo(double xCtrl, double yCtrl, | ||||
|                          double xTo,   double yTo); | ||||
|     void quadricCurveRel(double dxCtrl, double dyCtrl, | ||||
|                          double dxTo,   double dyTo); | ||||
|     void quadricCurveTo(double xTo, double yTo); | ||||
|     void quadricCurveRel(double dxTo, double dyTo); | ||||
|  | ||||
|     void cubicCurveTo(double xCtrl1, double yCtrl1, | ||||
|                       double xCtrl2, double yCtrl2, | ||||
|                       double xTo,    double yTo); | ||||
|  | ||||
|     void cubicCurveRel(double dxCtrl1, double dyCtrl1, | ||||
|                        double dxCtrl2, double dyCtrl2, | ||||
|                        double dxTo,    double dyTo); | ||||
|  | ||||
|     void cubicCurveTo(double xCtrl2, double yCtrl2, | ||||
|                       double xTo,    double yTo); | ||||
|  | ||||
|     void cubicCurveRel(double xCtrl2, double yCtrl2, | ||||
|                        double xTo,    double yTo); | ||||
|  | ||||
|     void addEllipse(double cx, double cy, double rx, double ry, Direction dir); | ||||
| 	void text(double x, double y, const std::string& text); | ||||
|     void closePolygon(); | ||||
|  | ||||
|     void drawPath(DrawPathFlag flag = FillAndStroke); | ||||
|     void drawPathNoTransform(DrawPathFlag flag = FillAndStroke); | ||||
|  | ||||
|  | ||||
|     // Image Transformations | ||||
|     //----------------------- | ||||
|     void imageFilter(ImageFilter f); | ||||
|     ImageFilter imageFilter() const; | ||||
|  | ||||
|     void imageResample(ImageResample f); | ||||
|     ImageResample imageResample() const; | ||||
|  | ||||
|     void transformImage(const Image& img, | ||||
|                            int imgX1,    int imgY1,    int imgX2,    int imgY2, | ||||
|                         double dstX1, double dstY1, double dstX2, double dstY2); | ||||
|  | ||||
|     void transformImage(const Image& img, | ||||
|                         double dstX1, double dstY1, double dstX2, double dstY2); | ||||
|  | ||||
|     void transformImage(const Image& img, | ||||
|                         int imgX1, int imgY1, int imgX2, int imgY2, | ||||
|                         const double* parallelogram); | ||||
|  | ||||
|     void transformImage(const Image& img, const double* parallelogram); | ||||
|  | ||||
|  | ||||
|     void transformImagePath(const Image& img, | ||||
|                                int imgX1,    int imgY1,    int imgX2,    int imgY2, | ||||
|                             double dstX1, double dstY1, double dstX2, double dstY2); | ||||
|  | ||||
|     void transformImagePath(const Image& img, | ||||
|                             double dstX1, double dstY1, double dstX2, double dstY2); | ||||
|  | ||||
|     void transformImagePath(const Image& img, | ||||
|                             int imgX1, int imgY1, int imgX2, int imgY2, | ||||
|                             const double* parallelogram); | ||||
|  | ||||
|     void transformImagePath(const Image& img, const double* parallelogram); | ||||
|  | ||||
|  | ||||
|     // Image Blending (no transformations available) | ||||
|     void blendImage(Image& img, | ||||
|                     int imgX1, int imgY1, int imgX2, int imgY2, | ||||
|                     double dstX, double dstY, unsigned alpha=255); | ||||
|     void blendImage(Image& img, double dstX, double dstY, unsigned alpha=255); | ||||
|  | ||||
|  | ||||
|     // Copy image directly, together with alpha-channel | ||||
|     void copyImage(Image& img, | ||||
|                    int imgX1, int imgY1, int imgX2, int imgY2, | ||||
|                    double dstX, double dstY); | ||||
|     void copyImage(Image& img, double dstX, double dstY); | ||||
|  | ||||
|  | ||||
|     // Auxiliary | ||||
|     //----------------------- | ||||
|     static double pi() { return agg::pi; } | ||||
|     static double deg2Rad(double v) { return v * agg::pi / 180.0; } | ||||
|     static double rad2Deg(double v) { return v * 180.0 / agg::pi; } | ||||
|  | ||||
| private: | ||||
|     void render(bool fillColor); | ||||
| #ifdef AGG_USE_FONTS | ||||
|     void render(FontRasterizer& ras, FontScanline& sl); | ||||
| #endif // AGG_USE_FONTS | ||||
|  | ||||
|     void addLine(double x1, double y1, double x2, double y2); | ||||
|     void updateRasterizerGamma(); | ||||
|     void renderImage(const Image& img, int x1, int y1, int x2, int y2, const double* parl); | ||||
|  | ||||
|     agg::rendering_buffer           m_rbuf; | ||||
|     PixFormat                       m_pixFormat; | ||||
|     PixFormatComp                   m_pixFormatComp; | ||||
|     PixFormatPre                    m_pixFormatPre; | ||||
|     PixFormatCompPre                m_pixFormatCompPre; | ||||
|     RendererBase                    m_renBase; | ||||
|     RendererBaseComp                m_renBaseComp; | ||||
|     RendererBasePre                 m_renBasePre; | ||||
|     RendererBaseCompPre             m_renBaseCompPre; | ||||
|     RendererSolid                   m_renSolid; | ||||
|     RendererSolidComp               m_renSolidComp; | ||||
|  | ||||
|     SpanAllocator                   m_allocator; | ||||
|     RectD                           m_clipBox; | ||||
|  | ||||
|     BlendMode                       m_blendMode; | ||||
|     BlendMode                       m_imageBlendMode; | ||||
|     Color                           m_imageBlendColor; | ||||
|  | ||||
|     agg::scanline_u8                m_scanline; | ||||
|     agg::rasterizer_scanline_aa<>   m_rasterizer; | ||||
|  | ||||
|     double                          m_masterAlpha; | ||||
|     double                          m_antiAliasGamma; | ||||
|  | ||||
|     Color                           m_fillColor; | ||||
|     Color                           m_lineColor; | ||||
|     GradientArray                   m_fillGradient; | ||||
|     GradientArray                   m_lineGradient; | ||||
|  | ||||
|     LineCap                         m_lineCap; | ||||
|     LineJoin                        m_lineJoin; | ||||
|  | ||||
|     Gradient                        m_fillGradientFlag; | ||||
|     Gradient                        m_lineGradientFlag; | ||||
|     agg::trans_affine               m_fillGradientMatrix; | ||||
|     agg::trans_affine               m_lineGradientMatrix; | ||||
|     double                          m_fillGradientD1; | ||||
|     double                          m_lineGradientD1; | ||||
|     double                          m_fillGradientD2; | ||||
|     double                          m_lineGradientD2; | ||||
|  | ||||
|     TextAlignment                   m_textAlignment; | ||||
|     double                          m_textSizeX; | ||||
|     double                          m_textSizeY; | ||||
|  | ||||
|     ImageFilter                     m_imageFilter; | ||||
|     ImageResample                   m_imageResample; | ||||
|     agg::image_filter_lut           m_imageFilterLut; | ||||
|  | ||||
|     agg::span_interpolator_linear<> m_fillGradientInterpolator; | ||||
|     agg::span_interpolator_linear<> m_lineGradientInterpolator; | ||||
|  | ||||
|     agg::gradient_x                 m_linearGradientFunction; | ||||
|     agg::gradient_circle            m_radialGradientFunction; | ||||
|  | ||||
|     double                          m_lineWidth; | ||||
|     bool                            m_evenOddFlag; | ||||
|  | ||||
|     agg::path_storage               m_path; | ||||
|     agg::trans_affine               m_transform; | ||||
|  | ||||
|     ConvCurve                       m_convCurve; | ||||
|     ConvStroke                      m_convStroke; | ||||
|  | ||||
|     PathTransform                   m_pathTransform; | ||||
|     StrokeTransform                 m_strokeTransform; | ||||
|  | ||||
| #ifdef AGG_USE_FONTS | ||||
| #ifndef AGG2D_USE_FREETYPE | ||||
|     HDC                             m_fontDC; | ||||
| #endif | ||||
|     FontEngine                      m_fontEngine; | ||||
|     FontCacheManager                m_fontCacheManager; | ||||
| #endif | ||||
| }; | ||||
|  | ||||
|  | ||||
| inline bool operator == (const Agg2D::Color& c1, const Agg2D::Color& c2) | ||||
| { | ||||
|    return c1.r == c2.r && c1.g == c2.g && c1.b == c2.b && c1.a == c2.a; | ||||
| } | ||||
|  | ||||
| inline bool operator != (const Agg2D::Color& c1, const Agg2D::Color& c2) | ||||
| { | ||||
|    return !(c1 == c2); | ||||
| } | ||||
|  | ||||
|  | ||||
| #endif | ||||
|  | ||||
|  | ||||
							
								
								
									
										499
									
								
								dep/agg/include/agg_alpha_mask_u8.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										499
									
								
								dep/agg/include/agg_alpha_mask_u8.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,499 @@ | ||||
| //---------------------------------------------------------------------------- | ||||
| // Anti-Grain Geometry - Version 2.4 | ||||
| // Copyright (C) 2002-2005 Maxim Shemanarev (http://www.antigrain.com) | ||||
| // | ||||
| // Permission to copy, use, modify, sell and distribute this software  | ||||
| // is granted provided this copyright notice appears in all copies.  | ||||
| // This software is provided "as is" without express or implied | ||||
| // warranty, and with no claim as to its suitability for any purpose. | ||||
| // | ||||
| //---------------------------------------------------------------------------- | ||||
| // Contact: mcseem@antigrain.com | ||||
| //          mcseemagg@yahoo.com | ||||
| //          http://www.antigrain.com | ||||
| //---------------------------------------------------------------------------- | ||||
| // | ||||
| // scanline_u8 class | ||||
| // | ||||
| //---------------------------------------------------------------------------- | ||||
| #ifndef AGG_ALPHA_MASK_U8_INCLUDED | ||||
| #define AGG_ALPHA_MASK_U8_INCLUDED | ||||
|  | ||||
| #include <cstring> | ||||
| #include "agg_basics.h" | ||||
| #include "agg_rendering_buffer.h" | ||||
|  | ||||
| namespace agg | ||||
| { | ||||
|     //===================================================one_component_mask_u8 | ||||
|     struct one_component_mask_u8 | ||||
|     { | ||||
|         static unsigned calculate(const int8u* p) { return *p; } | ||||
|     }; | ||||
|      | ||||
|  | ||||
|     //=====================================================rgb_to_gray_mask_u8 | ||||
|     template<unsigned R, unsigned G, unsigned B> | ||||
|     struct rgb_to_gray_mask_u8 | ||||
|     { | ||||
|         static unsigned calculate(const int8u* p)  | ||||
|         {  | ||||
|             return (p[R]*77 + p[G]*150 + p[B]*29) >> 8;  | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     //==========================================================alpha_mask_u8 | ||||
|     template<unsigned Step=1, unsigned Offset=0, class MaskF=one_component_mask_u8> | ||||
|     class alpha_mask_u8 | ||||
|     { | ||||
|     public: | ||||
|         typedef int8u cover_type; | ||||
|         typedef alpha_mask_u8<Step, Offset, MaskF> self_type; | ||||
|         enum cover_scale_e | ||||
|         {  | ||||
|             cover_shift = 8, | ||||
|             cover_none  = 0, | ||||
|             cover_full  = 255 | ||||
|         }; | ||||
|  | ||||
|         alpha_mask_u8() : m_rbuf(0) {} | ||||
|         explicit alpha_mask_u8(rendering_buffer& rbuf) : m_rbuf(&rbuf) {} | ||||
|  | ||||
|         void attach(rendering_buffer& rbuf) { m_rbuf = &rbuf; } | ||||
|  | ||||
|         MaskF& mask_function() { return m_mask_function; } | ||||
|         const MaskF& mask_function() const { return m_mask_function; } | ||||
|  | ||||
|          | ||||
|         //-------------------------------------------------------------------- | ||||
|         cover_type pixel(int x, int y) const | ||||
|         { | ||||
|             if(x >= 0 && y >= 0 &&  | ||||
|                x < (int)m_rbuf->width() &&  | ||||
|                y < (int)m_rbuf->height()) | ||||
|             { | ||||
|                 return (cover_type)m_mask_function.calculate( | ||||
|                                         m_rbuf->row_ptr(y) + x * Step + Offset); | ||||
|             } | ||||
|             return 0; | ||||
|         } | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         cover_type combine_pixel(int x, int y, cover_type val) const | ||||
|         { | ||||
|             if(x >= 0 && y >= 0 &&  | ||||
|                x < (int)m_rbuf->width() &&  | ||||
|                y < (int)m_rbuf->height()) | ||||
|             { | ||||
|                 return (cover_type)((cover_full + val *  | ||||
|                                      m_mask_function.calculate( | ||||
|                                         m_rbuf->row_ptr(y) + x * Step + Offset)) >>  | ||||
|                                      cover_shift); | ||||
|             } | ||||
|             return 0; | ||||
|         } | ||||
|  | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         void fill_hspan(int x, int y, cover_type* dst, int num_pix) const | ||||
|         { | ||||
|             int xmax = m_rbuf->width() - 1; | ||||
|             int ymax = m_rbuf->height() - 1; | ||||
|  | ||||
|             int count = num_pix; | ||||
|             cover_type* covers = dst; | ||||
|  | ||||
|             if(y < 0 || y > ymax) | ||||
|             { | ||||
|                 std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if(x < 0) | ||||
|             { | ||||
|                 count += x; | ||||
|                 if(count <= 0)  | ||||
|                 { | ||||
|                     std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                     return; | ||||
|                 } | ||||
|                 std::memset(covers, 0, -x * sizeof(cover_type)); | ||||
|                 covers -= x; | ||||
|                 x = 0; | ||||
|             } | ||||
|  | ||||
|             if(x + count > xmax) | ||||
|             { | ||||
|                 int rest = x + count - xmax - 1; | ||||
|                 count -= rest; | ||||
|                 if(count <= 0)  | ||||
|                 { | ||||
|                     std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                     return; | ||||
|                 } | ||||
|                 std::memset(covers + count, 0, rest * sizeof(cover_type)); | ||||
|             } | ||||
|  | ||||
|             const int8u* mask = m_rbuf->row_ptr(y) + x * Step + Offset; | ||||
|             do | ||||
|             { | ||||
|                 *covers++ = (cover_type)m_mask_function.calculate(mask); | ||||
|                 mask += Step; | ||||
|             } | ||||
|             while(--count); | ||||
|         } | ||||
|  | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         void combine_hspan(int x, int y, cover_type* dst, int num_pix) const | ||||
|         { | ||||
|             int xmax = m_rbuf->width() - 1; | ||||
|             int ymax = m_rbuf->height() - 1; | ||||
|  | ||||
|             int count = num_pix; | ||||
|             cover_type* covers = dst; | ||||
|  | ||||
|             if(y < 0 || y > ymax) | ||||
|             { | ||||
|                 std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if(x < 0) | ||||
|             { | ||||
|                 count += x; | ||||
|                 if(count <= 0)  | ||||
|                 { | ||||
|                     std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                     return; | ||||
|                 } | ||||
|                 std::memset(covers, 0, -x * sizeof(cover_type)); | ||||
|                 covers -= x; | ||||
|                 x = 0; | ||||
|             } | ||||
|  | ||||
|             if(x + count > xmax) | ||||
|             { | ||||
|                 int rest = x + count - xmax - 1; | ||||
|                 count -= rest; | ||||
|                 if(count <= 0)  | ||||
|                 { | ||||
|                     std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                     return; | ||||
|                 } | ||||
|                 std::memset(covers + count, 0, rest * sizeof(cover_type)); | ||||
|             } | ||||
|  | ||||
|             const int8u* mask = m_rbuf->row_ptr(y) + x * Step + Offset; | ||||
|             do | ||||
|             { | ||||
|                 *covers = (cover_type)((cover_full + (*covers) *  | ||||
|                                        m_mask_function.calculate(mask)) >>  | ||||
|                                        cover_shift); | ||||
|                 ++covers; | ||||
|                 mask += Step; | ||||
|             } | ||||
|             while(--count); | ||||
|         } | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         void fill_vspan(int x, int y, cover_type* dst, int num_pix) const | ||||
|         { | ||||
|             int xmax = m_rbuf->width() - 1; | ||||
|             int ymax = m_rbuf->height() - 1; | ||||
|  | ||||
|             int count = num_pix; | ||||
|             cover_type* covers = dst; | ||||
|  | ||||
|             if(x < 0 || x > xmax) | ||||
|             { | ||||
|                 std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if(y < 0) | ||||
|             { | ||||
|                 count += y; | ||||
|                 if(count <= 0)  | ||||
|                 { | ||||
|                     std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                     return; | ||||
|                 } | ||||
|                 std::memset(covers, 0, -y * sizeof(cover_type)); | ||||
|                 covers -= y; | ||||
|                 y = 0; | ||||
|             } | ||||
|  | ||||
|             if(y + count > ymax) | ||||
|             { | ||||
|                 int rest = y + count - ymax - 1; | ||||
|                 count -= rest; | ||||
|                 if(count <= 0)  | ||||
|                 { | ||||
|                     std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                     return; | ||||
|                 } | ||||
|                 std::memset(covers + count, 0, rest * sizeof(cover_type)); | ||||
|             } | ||||
|  | ||||
|             const int8u* mask = m_rbuf->row_ptr(y) + x * Step + Offset; | ||||
|             do | ||||
|             { | ||||
|                 *covers++ = (cover_type)m_mask_function.calculate(mask); | ||||
|                 mask += m_rbuf->stride(); | ||||
|             } | ||||
|             while(--count); | ||||
|         } | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         void combine_vspan(int x, int y, cover_type* dst, int num_pix) const | ||||
|         { | ||||
|             int xmax = m_rbuf->width() - 1; | ||||
|             int ymax = m_rbuf->height() - 1; | ||||
|  | ||||
|             int count = num_pix; | ||||
|             cover_type* covers = dst; | ||||
|  | ||||
|             if(x < 0 || x > xmax) | ||||
|             { | ||||
|                 std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                 return; | ||||
|             } | ||||
|  | ||||
|             if(y < 0) | ||||
|             { | ||||
|                 count += y; | ||||
|                 if(count <= 0)  | ||||
|                 { | ||||
|                     std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                     return; | ||||
|                 } | ||||
|                 std::memset(covers, 0, -y * sizeof(cover_type)); | ||||
|                 covers -= y; | ||||
|                 y = 0; | ||||
|             } | ||||
|  | ||||
|             if(y + count > ymax) | ||||
|             { | ||||
|                 int rest = y + count - ymax - 1; | ||||
|                 count -= rest; | ||||
|                 if(count <= 0)  | ||||
|                 { | ||||
|                     std::memset(dst, 0, num_pix * sizeof(cover_type)); | ||||
|                     return; | ||||
|                 } | ||||
|                 std::memset(covers + count, 0, rest * sizeof(cover_type)); | ||||
|             } | ||||
|  | ||||
|             const int8u* mask = m_rbuf->row_ptr(y) + x * Step + Offset; | ||||
|             do | ||||
|             { | ||||
|                 *covers = (cover_type)((cover_full + (*covers) *  | ||||
|                                        m_mask_function.calculate(mask)) >>  | ||||
|                                        cover_shift); | ||||
|                 ++covers; | ||||
|                 mask += m_rbuf->stride(); | ||||
|             } | ||||
|             while(--count); | ||||
|         } | ||||
|  | ||||
|  | ||||
|     private: | ||||
|         alpha_mask_u8(const self_type&); | ||||
|         const self_type& operator = (const self_type&); | ||||
|  | ||||
|         rendering_buffer* m_rbuf; | ||||
|         MaskF             m_mask_function; | ||||
|     }; | ||||
|      | ||||
|  | ||||
|     typedef alpha_mask_u8<1, 0> alpha_mask_gray8;   //----alpha_mask_gray8 | ||||
|  | ||||
|     typedef alpha_mask_u8<3, 0> alpha_mask_rgb24r;  //----alpha_mask_rgb24r | ||||
|     typedef alpha_mask_u8<3, 1> alpha_mask_rgb24g;  //----alpha_mask_rgb24g | ||||
|     typedef alpha_mask_u8<3, 2> alpha_mask_rgb24b;  //----alpha_mask_rgb24b | ||||
|  | ||||
|     typedef alpha_mask_u8<3, 2> alpha_mask_bgr24r;  //----alpha_mask_bgr24r | ||||
|     typedef alpha_mask_u8<3, 1> alpha_mask_bgr24g;  //----alpha_mask_bgr24g | ||||
|     typedef alpha_mask_u8<3, 0> alpha_mask_bgr24b;  //----alpha_mask_bgr24b | ||||
|  | ||||
|     typedef alpha_mask_u8<4, 0> alpha_mask_rgba32r; //----alpha_mask_rgba32r | ||||
|     typedef alpha_mask_u8<4, 1> alpha_mask_rgba32g; //----alpha_mask_rgba32g | ||||
|     typedef alpha_mask_u8<4, 2> alpha_mask_rgba32b; //----alpha_mask_rgba32b | ||||
|     typedef alpha_mask_u8<4, 3> alpha_mask_rgba32a; //----alpha_mask_rgba32a | ||||
|  | ||||
|     typedef alpha_mask_u8<4, 1> alpha_mask_argb32r; //----alpha_mask_argb32r | ||||
|     typedef alpha_mask_u8<4, 2> alpha_mask_argb32g; //----alpha_mask_argb32g | ||||
|     typedef alpha_mask_u8<4, 3> alpha_mask_argb32b; //----alpha_mask_argb32b | ||||
|     typedef alpha_mask_u8<4, 0> alpha_mask_argb32a; //----alpha_mask_argb32a | ||||
|  | ||||
|     typedef alpha_mask_u8<4, 2> alpha_mask_bgra32r; //----alpha_mask_bgra32r | ||||
|     typedef alpha_mask_u8<4, 1> alpha_mask_bgra32g; //----alpha_mask_bgra32g | ||||
|     typedef alpha_mask_u8<4, 0> alpha_mask_bgra32b; //----alpha_mask_bgra32b | ||||
|     typedef alpha_mask_u8<4, 3> alpha_mask_bgra32a; //----alpha_mask_bgra32a | ||||
|  | ||||
|     typedef alpha_mask_u8<4, 3> alpha_mask_abgr32r; //----alpha_mask_abgr32r | ||||
|     typedef alpha_mask_u8<4, 2> alpha_mask_abgr32g; //----alpha_mask_abgr32g | ||||
|     typedef alpha_mask_u8<4, 1> alpha_mask_abgr32b; //----alpha_mask_abgr32b | ||||
|     typedef alpha_mask_u8<4, 0> alpha_mask_abgr32a; //----alpha_mask_abgr32a | ||||
|  | ||||
|     typedef alpha_mask_u8<3, 0, rgb_to_gray_mask_u8<0, 1, 2> > alpha_mask_rgb24gray;  //----alpha_mask_rgb24gray | ||||
|     typedef alpha_mask_u8<3, 0, rgb_to_gray_mask_u8<2, 1, 0> > alpha_mask_bgr24gray;  //----alpha_mask_bgr24gray | ||||
|     typedef alpha_mask_u8<4, 0, rgb_to_gray_mask_u8<0, 1, 2> > alpha_mask_rgba32gray; //----alpha_mask_rgba32gray | ||||
|     typedef alpha_mask_u8<4, 1, rgb_to_gray_mask_u8<0, 1, 2> > alpha_mask_argb32gray; //----alpha_mask_argb32gray | ||||
|     typedef alpha_mask_u8<4, 0, rgb_to_gray_mask_u8<2, 1, 0> > alpha_mask_bgra32gray; //----alpha_mask_bgra32gray | ||||
|     typedef alpha_mask_u8<4, 1, rgb_to_gray_mask_u8<2, 1, 0> > alpha_mask_abgr32gray; //----alpha_mask_abgr32gray | ||||
|  | ||||
|  | ||||
|  | ||||
|     //==========================================================amask_no_clip_u8 | ||||
|     template<unsigned Step=1, unsigned Offset=0, class MaskF=one_component_mask_u8> | ||||
|     class amask_no_clip_u8 | ||||
|     { | ||||
|     public: | ||||
|         typedef int8u cover_type; | ||||
|         typedef amask_no_clip_u8<Step, Offset, MaskF> self_type; | ||||
|         enum cover_scale_e | ||||
|         {  | ||||
|             cover_shift = 8, | ||||
|             cover_none  = 0, | ||||
|             cover_full  = 255 | ||||
|         }; | ||||
|  | ||||
|         amask_no_clip_u8() : m_rbuf(0) {} | ||||
|         explicit amask_no_clip_u8(rendering_buffer& rbuf) : m_rbuf(&rbuf) {} | ||||
|  | ||||
|         void attach(rendering_buffer& rbuf) { m_rbuf = &rbuf; } | ||||
|  | ||||
|         MaskF& mask_function() { return m_mask_function; } | ||||
|         const MaskF& mask_function() const { return m_mask_function; } | ||||
|  | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         cover_type pixel(int x, int y) const | ||||
|         { | ||||
|             return (cover_type)m_mask_function.calculate( | ||||
|                                    m_rbuf->row_ptr(y) + x * Step + Offset); | ||||
|         } | ||||
|  | ||||
|          | ||||
|         //-------------------------------------------------------------------- | ||||
|         cover_type combine_pixel(int x, int y, cover_type val) const | ||||
|         { | ||||
|             return (cover_type)((cover_full + val *  | ||||
|                                  m_mask_function.calculate( | ||||
|                                     m_rbuf->row_ptr(y) + x * Step + Offset)) >>  | ||||
|                                  cover_shift); | ||||
|         } | ||||
|  | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         void fill_hspan(int x, int y, cover_type* dst, int num_pix) const | ||||
|         { | ||||
|             const int8u* mask = m_rbuf->row_ptr(y) + x * Step + Offset; | ||||
|             do | ||||
|             { | ||||
|                 *dst++ = (cover_type)m_mask_function.calculate(mask); | ||||
|                 mask += Step; | ||||
|             } | ||||
|             while(--num_pix); | ||||
|         } | ||||
|  | ||||
|  | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         void combine_hspan(int x, int y, cover_type* dst, int num_pix) const | ||||
|         { | ||||
|             const int8u* mask = m_rbuf->row_ptr(y) + x * Step + Offset; | ||||
|             do | ||||
|             { | ||||
|                 *dst = (cover_type)((cover_full + (*dst) *  | ||||
|                                     m_mask_function.calculate(mask)) >>  | ||||
|                                     cover_shift); | ||||
|                 ++dst; | ||||
|                 mask += Step; | ||||
|             } | ||||
|             while(--num_pix); | ||||
|         } | ||||
|  | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         void fill_vspan(int x, int y, cover_type* dst, int num_pix) const | ||||
|         { | ||||
|             const int8u* mask = m_rbuf->row_ptr(y) + x * Step + Offset; | ||||
|             do | ||||
|             { | ||||
|                 *dst++ = (cover_type)m_mask_function.calculate(mask); | ||||
|                 mask += m_rbuf->stride(); | ||||
|             } | ||||
|             while(--num_pix); | ||||
|         } | ||||
|  | ||||
|  | ||||
|         //-------------------------------------------------------------------- | ||||
|         void combine_vspan(int x, int y, cover_type* dst, int num_pix) const | ||||
|         { | ||||
|             const int8u* mask = m_rbuf->row_ptr(y) + x * Step + Offset; | ||||
|             do | ||||
|             { | ||||
|                 *dst = (cover_type)((cover_full + (*dst) *  | ||||
|                                     m_mask_function.calculate(mask)) >>  | ||||
|                                     cover_shift); | ||||
|                 ++dst; | ||||
|                 mask += m_rbuf->stride(); | ||||
|             } | ||||
|             while(--num_pix); | ||||
|         } | ||||
|  | ||||
|     private: | ||||
|         amask_no_clip_u8(const self_type&); | ||||
|         const self_type& operator = (const self_type&); | ||||
|  | ||||
|         rendering_buffer* m_rbuf; | ||||
|         MaskF             m_mask_function; | ||||
|     }; | ||||
|      | ||||
|  | ||||
|     typedef amask_no_clip_u8<1, 0> amask_no_clip_gray8;   //----amask_no_clip_gray8 | ||||
|  | ||||
|     typedef amask_no_clip_u8<3, 0> amask_no_clip_rgb24r;  //----amask_no_clip_rgb24r | ||||
|     typedef amask_no_clip_u8<3, 1> amask_no_clip_rgb24g;  //----amask_no_clip_rgb24g | ||||
|     typedef amask_no_clip_u8<3, 2> amask_no_clip_rgb24b;  //----amask_no_clip_rgb24b | ||||
|  | ||||
|     typedef amask_no_clip_u8<3, 2> amask_no_clip_bgr24r;  //----amask_no_clip_bgr24r | ||||
|     typedef amask_no_clip_u8<3, 1> amask_no_clip_bgr24g;  //----amask_no_clip_bgr24g | ||||
|     typedef amask_no_clip_u8<3, 0> amask_no_clip_bgr24b;  //----amask_no_clip_bgr24b | ||||
|  | ||||
|     typedef amask_no_clip_u8<4, 0> amask_no_clip_rgba32r; //----amask_no_clip_rgba32r | ||||
|     typedef amask_no_clip_u8<4, 1> amask_no_clip_rgba32g; //----amask_no_clip_rgba32g | ||||
|     typedef amask_no_clip_u8<4, 2> amask_no_clip_rgba32b; //----amask_no_clip_rgba32b | ||||
|     typedef amask_no_clip_u8<4, 3> amask_no_clip_rgba32a; //----amask_no_clip_rgba32a | ||||
|  | ||||
|     typedef amask_no_clip_u8<4, 1> amask_no_clip_argb32r; //----amask_no_clip_argb32r | ||||
|     typedef amask_no_clip_u8<4, 2> amask_no_clip_argb32g; //----amask_no_clip_argb32g | ||||
|     typedef amask_no_clip_u8<4, 3> amask_no_clip_argb32b; //----amask_no_clip_argb32b | ||||
|     typedef amask_no_clip_u8<4, 0> amask_no_clip_argb32a; //----amask_no_clip_argb32a | ||||
|  | ||||
|     typedef amask_no_clip_u8<4, 2> amask_no_clip_bgra32r; //----amask_no_clip_bgra32r | ||||
|     typedef amask_no_clip_u8<4, 1> amask_no_clip_bgra32g; //----amask_no_clip_bgra32g | ||||
|     typedef amask_no_clip_u8<4, 0> amask_no_clip_bgra32b; //----amask_no_clip_bgra32b | ||||
|     typedef amask_no_clip_u8<4, 3> amask_no_clip_bgra32a; //----amask_no_clip_bgra32a | ||||
|  | ||||
|     typedef amask_no_clip_u8<4, 3> amask_no_clip_abgr32r; //----amask_no_clip_abgr32r | ||||
|     typedef amask_no_clip_u8<4, 2> amask_no_clip_abgr32g; //----amask_no_clip_abgr32g | ||||
|     typedef amask_no_clip_u8<4, 1> amask_no_clip_abgr32b; //----amask_no_clip_abgr32b | ||||
|     typedef amask_no_clip_u8<4, 0> amask_no_clip_abgr32a; //----amask_no_clip_abgr32a | ||||
|  | ||||
|     typedef amask_no_clip_u8<3, 0, rgb_to_gray_mask_u8<0, 1, 2> > amask_no_clip_rgb24gray;  //----amask_no_clip_rgb24gray | ||||
|     typedef amask_no_clip_u8<3, 0, rgb_to_gray_mask_u8<2, 1, 0> > amask_no_clip_bgr24gray;  //----amask_no_clip_bgr24gray | ||||
|     typedef amask_no_clip_u8<4, 0, rgb_to_gray_mask_u8<0, 1, 2> > amask_no_clip_rgba32gray; //----amask_no_clip_rgba32gray | ||||
|     typedef amask_no_clip_u8<4, 1, rgb_to_gray_mask_u8<0, 1, 2> > amask_no_clip_argb32gray; //----amask_no_clip_argb32gray | ||||
|     typedef amask_no_clip_u8<4, 0, rgb_to_gray_mask_u8<2, 1, 0> > amask_no_clip_bgra32gray; //----amask_no_clip_bgra32gray | ||||
|     typedef amask_no_clip_u8<4, 1, rgb_to_gray_mask_u8<2, 1, 0> > amask_no_clip_abgr32gray; //----amask_no_clip_abgr32gray | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										73
									
								
								dep/agg/include/agg_arc.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								dep/agg/include/agg_arc.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,73 @@ | ||||
| //---------------------------------------------------------------------------- | ||||
| // Anti-Grain Geometry - Version 2.4 | ||||
| // Copyright (C) 2002-2005 Maxim Shemanarev (http://www.antigrain.com) | ||||
| // | ||||
| // Permission to copy, use, modify, sell and distribute this software  | ||||
| // is granted provided this copyright notice appears in all copies.  | ||||
| // This software is provided "as is" without express or implied | ||||
| // warranty, and with no claim as to its suitability for any purpose. | ||||
| // | ||||
| //---------------------------------------------------------------------------- | ||||
| // Contact: mcseem@antigrain.com | ||||
| //          mcseemagg@yahoo.com | ||||
| //          http://www.antigrain.com | ||||
| //---------------------------------------------------------------------------- | ||||
| // | ||||
| // Arc vertex generator | ||||
| // | ||||
| //---------------------------------------------------------------------------- | ||||
|  | ||||
| #ifndef AGG_ARC_INCLUDED | ||||
| #define AGG_ARC_INCLUDED | ||||
|  | ||||
| #include "agg_basics.h" | ||||
|  | ||||
| namespace agg | ||||
| { | ||||
|  | ||||
|     //=====================================================================arc | ||||
|     // | ||||
|     // See Implementation agg_arc.cpp  | ||||
|     // | ||||
|     class arc | ||||
|     { | ||||
|     public: | ||||
|         arc() : m_scale(1.0), m_initialized(false) {} | ||||
|         arc(double x,  double y,  | ||||
|             double rx, double ry,  | ||||
|             double a1, double a2,  | ||||
|             bool ccw=true); | ||||
|  | ||||
|         void init(double x,  double y,  | ||||
|                   double rx, double ry,  | ||||
|                   double a1, double a2,  | ||||
|                   bool ccw=true); | ||||
|  | ||||
|         void approximation_scale(double s); | ||||
|         double approximation_scale() const { return m_scale;  } | ||||
|  | ||||
|         void rewind(unsigned); | ||||
|         unsigned vertex(double* x, double* y); | ||||
|  | ||||
|     private: | ||||
|         void normalize(double a1, double a2, bool ccw); | ||||
|  | ||||
|         double   m_x; | ||||
|         double   m_y; | ||||
|         double   m_rx; | ||||
|         double   m_ry; | ||||
|         double   m_angle; | ||||
|         double   m_start; | ||||
|         double   m_end; | ||||
|         double   m_scale; | ||||
|         double   m_da; | ||||
|         bool     m_ccw; | ||||
|         bool     m_initialized; | ||||
|         unsigned m_path_cmd; | ||||
|     }; | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
|  | ||||
| #endif | ||||
							
								
								
									
										1119
									
								
								dep/agg/include/agg_array.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1119
									
								
								dep/agg/include/agg_array.h
									
									
									
									
									
										Normal file
									
								
							
										
											
												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