Compare commits
	
		
			1 Commits
		
	
	
		
			9b3ab73bb1
			...
			refactor
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 9baa15050e | 
							
								
								
									
										452
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										452
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							| @@ -23,7 +23,7 @@ dependencies = [ | ||||
|  "alsa-sys", | ||||
|  "bitflags", | ||||
|  "libc", | ||||
|  "nix 0.20.0", | ||||
|  "nix 0.20.1", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| @@ -58,20 +58,11 @@ dependencies = [ | ||||
|  "winapi", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "ansi_term" | ||||
| version = "0.12.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "d52a9bb7ec0cf484c551830a7ce27bd20d67eac647e1befb56b0be4ee39a55d2" | ||||
| dependencies = [ | ||||
|  "winapi", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "anyhow" | ||||
| version = "1.0.44" | ||||
| version = "1.0.43" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "61604a8f862e1d5c3229fdd78f8b02c68dcf73a4c4b05fd636d12240aaa242c1" | ||||
| checksum = "28ae2b3dec75a406790005a200b1bd89785afc02517a00ca99ecfe093ee9e6cf" | ||||
|  | ||||
| [[package]] | ||||
| name = "arrayvec" | ||||
| @@ -85,7 +76,7 @@ version = "0.33.3+1.2.191" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "cc4f1d82f164f838ae413296d1131aa6fa79b917d25bebaa7033d25620c09219" | ||||
| dependencies = [ | ||||
|  "libloading 0.7.1", | ||||
|  "libloading 0.7.0", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| @@ -153,9 +144,9 @@ checksum = "46afbd2983a5d5a7bd740ccb198caf5b82f45c40c09c0eed36052d91cb92e719" | ||||
|  | ||||
| [[package]] | ||||
| name = "bitflags" | ||||
| version = "1.3.2" | ||||
| version = "1.2.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" | ||||
| checksum = "cf1de2fe8c75bc145a2f577add951f8134889b4795d47466a54a5c846d691693" | ||||
|  | ||||
| [[package]] | ||||
| name = "block" | ||||
| @@ -165,9 +156,9 @@ checksum = "0d8c1fef690941d3e7788d328517591fecc684c084084702d6ff1641e993699a" | ||||
|  | ||||
| [[package]] | ||||
| name = "bumpalo" | ||||
| version = "3.7.1" | ||||
| version = "3.7.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "d9df67f7bf9ef8498769f994239c45613ef0c5899415fb58e9add412d2c1a538" | ||||
| checksum = "9c59e7af012c713f529e7a3ee57ce9b31ddd858d4b512923602f74608b009631" | ||||
|  | ||||
| [[package]] | ||||
| name = "bytemuck" | ||||
| @@ -205,9 +196,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "cc" | ||||
| version = "1.0.71" | ||||
| version = "1.0.70" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "79c2681d6594606957bbb8631c4b90a7fcaaa72cdb714743a437b156d6a7eedd" | ||||
| checksum = "d26a6ce4b6a484fa3edb70f7efa6fc430fd2b87285fe8b84304fd0936faa0dc0" | ||||
| dependencies = [ | ||||
|  "jobserver", | ||||
| ] | ||||
| @@ -245,18 +236,6 @@ version = "0.1.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "fd16c4719339c4530435d38e511904438d07cce7950afa3718a84ac36c10e89e" | ||||
|  | ||||
| [[package]] | ||||
| name = "chrono" | ||||
| version = "0.4.19" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "670ad68c9088c2a963aaa298cb369688cf3f9465ce5e2d4ca10e6e0098a1ce73" | ||||
| dependencies = [ | ||||
|  "libc", | ||||
|  "num-integer", | ||||
|  "num-traits", | ||||
|  "winapi", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "clang-sys" | ||||
| version = "1.2.2" | ||||
| @@ -265,7 +244,7 @@ checksum = "10612c0ec0e0a1ff0e97980647cb058a6e7aedb913d01d009c406b8b7d0b26ee" | ||||
| dependencies = [ | ||||
|  "glob", | ||||
|  "libc", | ||||
|  "libloading 0.7.1", | ||||
|  "libloading 0.7.0", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| @@ -274,7 +253,7 @@ version = "2.33.3" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "37e58ac78573c40708d45522f0d80fa2f01cc4f9b4e2bf749807255454312002" | ||||
| dependencies = [ | ||||
|  "ansi_term 0.11.0", | ||||
|  "ansi_term", | ||||
|  "atty", | ||||
|  "bitflags", | ||||
|  "strsim 0.8.0", | ||||
| @@ -298,7 +277,7 @@ dependencies = [ | ||||
|  "bitflags", | ||||
|  "block", | ||||
|  "cocoa-foundation", | ||||
|  "core-foundation 0.9.2", | ||||
|  "core-foundation 0.9.1", | ||||
|  "core-graphics 0.22.2", | ||||
|  "foreign-types", | ||||
|  "libc", | ||||
| @@ -313,7 +292,7 @@ checksum = "7ade49b65d560ca58c403a479bb396592b155c0185eada742ee323d1d68d6318" | ||||
| dependencies = [ | ||||
|  "bitflags", | ||||
|  "block", | ||||
|  "core-foundation 0.9.2", | ||||
|  "core-foundation 0.9.1", | ||||
|  "core-graphics-types", | ||||
|  "foreign-types", | ||||
|  "libc", | ||||
| @@ -368,11 +347,11 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "core-foundation" | ||||
| version = "0.9.2" | ||||
| version = "0.9.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "6888e10551bb93e424d8df1d07f1a8b4fceb0001a3a4b048bfc47554946f47b3" | ||||
| checksum = "0a89e2ae426ea83155dccf10c0fa6b1463ef6d5fcb44cee0b224a408fa640a62" | ||||
| dependencies = [ | ||||
|  "core-foundation-sys 0.8.3", | ||||
|  "core-foundation-sys 0.8.2", | ||||
|  "libc", | ||||
| ] | ||||
|  | ||||
| @@ -390,9 +369,9 @@ checksum = "b3a71ab494c0b5b860bdc8407ae08978052417070c2ced38573a9157ad75b8ac" | ||||
|  | ||||
| [[package]] | ||||
| name = "core-foundation-sys" | ||||
| version = "0.8.3" | ||||
| version = "0.8.2" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "5827cebf4670468b8772dd191856768aedcb1b0278a04f989f7766351917b9dc" | ||||
| checksum = "ea221b5284a47e40033bf9b66f35f984ec0ea2931eb03505246cd27a963f981b" | ||||
|  | ||||
| [[package]] | ||||
| name = "core-graphics" | ||||
| @@ -413,7 +392,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "269f35f69b542b80e736a20a89a05215c0ce80c2c03c514abb2e318b78379d86" | ||||
| dependencies = [ | ||||
|  "bitflags", | ||||
|  "core-foundation 0.9.2", | ||||
|  "core-foundation 0.9.1", | ||||
|  "core-graphics-types", | ||||
|  "foreign-types", | ||||
|  "libc", | ||||
| @@ -426,7 +405,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "3a68b68b3446082644c91ac778bf50cd4104bfb002b5a6a7c44cca5a2c70788b" | ||||
| dependencies = [ | ||||
|  "bitflags", | ||||
|  "core-foundation 0.9.2", | ||||
|  "core-foundation 0.9.1", | ||||
|  "foreign-types", | ||||
|  "libc", | ||||
| ] | ||||
| @@ -470,7 +449,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "98f45f0a21f617cd2c788889ef710b63f075c949259593ea09c826f1e47a2418" | ||||
| dependencies = [ | ||||
|  "alsa", | ||||
|  "core-foundation-sys 0.8.3", | ||||
|  "core-foundation-sys 0.8.2", | ||||
|  "coreaudio-rs", | ||||
|  "jni", | ||||
|  "js-sys", | ||||
| @@ -479,7 +458,7 @@ dependencies = [ | ||||
|  "mach 0.3.2", | ||||
|  "ndk 0.3.0", | ||||
|  "ndk-glue 0.3.0", | ||||
|  "nix 0.20.0", | ||||
|  "nix 0.20.1", | ||||
|  "oboe", | ||||
|  "parking_lot", | ||||
|  "stdweb 0.1.3", | ||||
| @@ -563,7 +542,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "2daefd788d1e96e0a9d66dee4b828b883509bc3ea9ce30665f04c3246372690c" | ||||
| dependencies = [ | ||||
|  "bitflags", | ||||
|  "libloading 0.7.1", | ||||
|  "libloading 0.7.0", | ||||
|  "winapi", | ||||
| ] | ||||
|  | ||||
| @@ -613,47 +592,6 @@ dependencies = [ | ||||
|  "syn", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "directories-next" | ||||
| version = "2.0.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "339ee130d97a610ea5a5872d2bbb130fdf68884ff09d3028b81bec8a1ac23bbc" | ||||
| dependencies = [ | ||||
|  "cfg-if 1.0.0", | ||||
|  "dirs-sys-next", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "dirs" | ||||
| version = "3.0.2" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "30baa043103c9d0c2a57cf537cc2f35623889dc0d405e6c3cccfadbc81c71309" | ||||
| dependencies = [ | ||||
|  "dirs-sys", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "dirs-sys" | ||||
| version = "0.3.6" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "03d86534ed367a67548dc68113a0f5db55432fdfbb6e6f9d77704397d95d5780" | ||||
| dependencies = [ | ||||
|  "libc", | ||||
|  "redox_users", | ||||
|  "winapi", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "dirs-sys-next" | ||||
| version = "0.1.2" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "4ebda144c4fe02d1f7ea1a7d9641b6fc6b580adcfa024ae48797ecdeb6825b4d" | ||||
| dependencies = [ | ||||
|  "libc", | ||||
|  "redox_users", | ||||
|  "winapi", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "discard" | ||||
| version = "1.0.4" | ||||
| @@ -681,7 +619,7 @@ version = "0.5.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "ac1b7517328c04c2aa68422fc60a41b92208182142ed04a25879c26c8f878794" | ||||
| dependencies = [ | ||||
|  "libloading 0.7.1", | ||||
|  "libloading 0.7.0", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| @@ -690,6 +628,12 @@ version = "1.2.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "9ea835d29036a4087793836fa931b08837ad5e957da9e23886b29586fb9b6650" | ||||
|  | ||||
| [[package]] | ||||
| name = "fixedbitset" | ||||
| version = "0.4.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "398ea4fabe40b9b0d885340a2a991a44c8a645624075ad966d21f88688e2b69e" | ||||
|  | ||||
| [[package]] | ||||
| name = "fnv" | ||||
| version = "1.0.7" | ||||
| @@ -727,28 +671,14 @@ dependencies = [ | ||||
|  "anyhow", | ||||
|  "bitfield", | ||||
|  "clap", | ||||
|  "directories-next", | ||||
|  "gilrs", | ||||
|  "pixels", | ||||
|  "rodio", | ||||
|  "rtrb", | ||||
|  "tracing", | ||||
|  "tracing-subscriber", | ||||
|  "winit", | ||||
|  "winit_input_helper", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "getrandom" | ||||
| version = "0.2.3" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "7fcd999463524c52659517fe2cea98493cfe485d10565e7b0fb07dbba7ad2753" | ||||
| dependencies = [ | ||||
|  "cfg-if 1.0.0", | ||||
|  "libc", | ||||
|  "wasi", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "gilrs" | ||||
| version = "0.8.1" | ||||
| @@ -773,7 +703,7 @@ dependencies = [ | ||||
|  "libc", | ||||
|  "libudev-sys", | ||||
|  "log", | ||||
|  "nix 0.20.0", | ||||
|  "nix 0.20.1", | ||||
|  "rusty-xinput", | ||||
|  "stdweb 0.4.20", | ||||
|  "uuid", | ||||
| @@ -801,9 +731,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "gpu-alloc" | ||||
| version = "0.5.2" | ||||
| version = "0.5.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "0e64cbb8d36508d3e19da95e56e196a84f674fc190881f2cc010000798838aa6" | ||||
| checksum = "ab8524eac5fc9d05625c891adf78fcf64dc0ee9f8d0882874b9f220f42b442bf" | ||||
| dependencies = [ | ||||
|  "bitflags", | ||||
|  "gpu-alloc-types", | ||||
| @@ -862,12 +792,6 @@ dependencies = [ | ||||
|  "libc", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "hexf-parse" | ||||
| version = "0.2.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "dfa686283ad6dd069f105e5ab091b04c62850d3e4cf5d67debad1933f55023df" | ||||
|  | ||||
| [[package]] | ||||
| name = "hound" | ||||
| version = "3.4.0" | ||||
| @@ -898,14 +822,11 @@ checksum = "90953f308a79fe6d62a4643e51f848fbfddcd05975a38e69fdf4ab86a7baf7ca" | ||||
|  | ||||
| [[package]] | ||||
| name = "instant" | ||||
| version = "0.1.11" | ||||
| version = "0.1.10" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "716d3d89f35ac6a34fd0eed635395f4c3b76fa889338a4632e5231a8684216bd" | ||||
| checksum = "bee0328b1209d157ef001c94dd85b4f8f64139adb0eac2659f4b08382b2f474d" | ||||
| dependencies = [ | ||||
|  "cfg-if 1.0.0", | ||||
|  "js-sys", | ||||
|  "wasm-bindgen", | ||||
|  "web-sys", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| @@ -955,9 +876,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "js-sys" | ||||
| version = "0.3.55" | ||||
| version = "0.3.54" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "7cc9ffccd38c451a86bf13657df244e9c3f37493cce8e5e21e940963777acc84" | ||||
| checksum = "1866b355d9c878e5e607473cbe3f63282c0b7aad2db1dbebf55076c686918254" | ||||
| dependencies = [ | ||||
|  "wasm-bindgen", | ||||
| ] | ||||
| @@ -969,7 +890,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "8c2352bd1d0bceb871cb9d40f24360c8133c11d7486b68b5381c1dd1a32015e3" | ||||
| dependencies = [ | ||||
|  "libc", | ||||
|  "libloading 0.7.1", | ||||
|  "libloading 0.7.0", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| @@ -997,9 +918,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "libc" | ||||
| version = "0.2.103" | ||||
| version = "0.2.101" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "dd8f7255a17a627354f321ef0055d63b898c6fb27eff628af4d1b66b7331edf6" | ||||
| checksum = "3cb00336871be5ed2c8ed44b60ae9959dc5b9f08539422ed43f09e34ecaeba21" | ||||
|  | ||||
| [[package]] | ||||
| name = "libloading" | ||||
| @@ -1013,9 +934,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "libloading" | ||||
| version = "0.7.1" | ||||
| version = "0.7.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "c0cf036d15402bea3c5d4de17b3fce76b3e4a56ebc1f577be0e7a72f7c607cf0" | ||||
| checksum = "6f84d96438c15fcd6c3f244c8fce01d1e2b9c6b5623e9c711dc9286d8fc92d6a" | ||||
| dependencies = [ | ||||
|  "cfg-if 1.0.0", | ||||
|  "winapi", | ||||
| @@ -1077,13 +998,10 @@ dependencies = [ | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "matchers" | ||||
| version = "0.0.1" | ||||
| name = "maybe-uninit" | ||||
| version = "2.0.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "f099785f7595cc4b4553a174ce30dd7589ef93391ff414dbb67f62392b9e0ce1" | ||||
| dependencies = [ | ||||
|  "regex-automata", | ||||
| ] | ||||
| checksum = "60302e4db3a61da70c0cb7991976248362f30319e88850c487b9b95bbf059e00" | ||||
|  | ||||
| [[package]] | ||||
| name = "memchr" | ||||
| @@ -1125,9 +1043,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "minimal-lexical" | ||||
| version = "0.1.4" | ||||
| version = "0.1.3" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "9c64630dcdd71f1a64c435f54885086a0de5d6a12d104d69b165fb7d5286d677" | ||||
| checksum = "0c835948974f68e0bd58636fc6c5b1fbff7b297e3046f11b3b3c18bbac012c6d" | ||||
|  | ||||
| [[package]] | ||||
| name = "minimp3" | ||||
| @@ -1185,18 +1103,17 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "naga" | ||||
| version = "0.7.1" | ||||
| version = "0.6.3" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "eda66d09f712e1f0a6ab436137da4fac312f78301f6d4ac7cb8bfe96e988734f" | ||||
| checksum = "8c5859e55c51da10b98e7a73068e0a0c5da7bbcae4fc38f86043d0c6d1b917cf" | ||||
| dependencies = [ | ||||
|  "bit-set", | ||||
|  "bitflags", | ||||
|  "codespan-reporting", | ||||
|  "fxhash", | ||||
|  "hexf-parse", | ||||
|  "indexmap", | ||||
|  "log", | ||||
|  "num-traits", | ||||
|  "petgraph", | ||||
|  "spirv", | ||||
|  "thiserror", | ||||
| ] | ||||
| @@ -1287,14 +1204,15 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "nix" | ||||
| version = "0.20.0" | ||||
| version = "0.20.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "fa9b4819da1bc61c0ea48b63b7bc8604064dd43013e7cc325df098d49cd7c18a" | ||||
| checksum = "df8e5e343312e7fbeb2a52139114e9e702991ef9c2aea6817ff2440b35647d56" | ||||
| dependencies = [ | ||||
|  "bitflags", | ||||
|  "cc", | ||||
|  "cfg-if 1.0.0", | ||||
|  "libc", | ||||
|  "memoffset", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| @@ -1338,16 +1256,6 @@ dependencies = [ | ||||
|  "syn", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "num-integer" | ||||
| version = "0.1.44" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "d2cc698a63b549a70bc047073d2949cce27cd1c7b0a4a862d08a8031bc2801db" | ||||
| dependencies = [ | ||||
|  "autocfg", | ||||
|  "num-traits", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "num-traits" | ||||
| version = "0.2.14" | ||||
| @@ -1373,7 +1281,7 @@ version = "0.5.4" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "486ea01961c4a818096de679a8b740b26d9033146ac5291b1c98557658f8cdd9" | ||||
| dependencies = [ | ||||
|  "proc-macro-crate 1.1.0", | ||||
|  "proc-macro-crate 1.0.0", | ||||
|  "proc-macro2", | ||||
|  "quote", | ||||
|  "syn", | ||||
| @@ -1483,16 +1391,20 @@ source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "d4fd5641d01c8f18a23da7b6fe29298ff4b55afcccdf78973b24cf3175fee32e" | ||||
|  | ||||
| [[package]] | ||||
| name = "pin-project-lite" | ||||
| version = "0.2.7" | ||||
| name = "petgraph" | ||||
| version = "0.6.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "8d31d11c69a6b52a174b42bdc0c30e5e11670f90788b2c471c31c1d17d449443" | ||||
| checksum = "4a13a2fa9d0b63e5f22328828741e523766fff0ee9e779316902290dff3f824f" | ||||
| dependencies = [ | ||||
|  "fixedbitset", | ||||
|  "indexmap", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "pixels" | ||||
| version = "0.7.0" | ||||
| version = "0.6.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "94ab1e297051c39cc7b7511e7e2b3ab151f14aff6a44e73bdde652b1e2190950" | ||||
| checksum = "6e794d4eab6519352618cca630ce4970a65576b9df54eca2068d4aad4b143903" | ||||
| dependencies = [ | ||||
|  "bytemuck", | ||||
|  "pollster", | ||||
| @@ -1504,9 +1416,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "pkg-config" | ||||
| version = "0.3.20" | ||||
| version = "0.3.19" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "7c9b1041b4387893b91ee6746cddfc28516aff326a3519fb2adf820932c5e6cb" | ||||
| checksum = "3831453b3449ceb48b6d9c7ad7c96d5ea673e9b470a1dc578c2ce6521230884c" | ||||
|  | ||||
| [[package]] | ||||
| name = "pollster" | ||||
| @@ -1525,9 +1437,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "proc-macro-crate" | ||||
| version = "1.1.0" | ||||
| version = "1.0.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "1ebace6889caf889b4d3f76becee12e90353f2b8c7d875534a71e5742f8f6f83" | ||||
| checksum = "41fdbd1df62156fbc5945f4762632564d7d038153091c3fcf1067f6aef7cff92" | ||||
| dependencies = [ | ||||
|  "thiserror", | ||||
|  "toml", | ||||
| @@ -1535,9 +1447,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "proc-macro2" | ||||
| version = "1.0.30" | ||||
| version = "1.0.29" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "edc3358ebc67bc8b7fa0c007f945b0b18226f78437d61bec735a9eb96b61ee70" | ||||
| checksum = "b9f5105d4fdaab20335ca9565e106a5d9b82b6219b5ba735731124ac6711d23d" | ||||
| dependencies = [ | ||||
|  "unicode-xid", | ||||
| ] | ||||
| @@ -1550,9 +1462,9 @@ checksum = "87dfd5592a8eed7e74f56ad7b125f8234763b805c30f0c7c95c486920026a6ec" | ||||
|  | ||||
| [[package]] | ||||
| name = "quote" | ||||
| version = "1.0.10" | ||||
| version = "1.0.9" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "38bc8cc6a5f2e3655e0899c1b848643b2562f853f114bfec7be120678e3ace05" | ||||
| checksum = "c3d0b9745dc2debf507c8422de05d7226cc1f0644216dfdfead988f9b1ab32a7" | ||||
| dependencies = [ | ||||
|  "proc-macro2", | ||||
| ] | ||||
| @@ -1581,16 +1493,6 @@ dependencies = [ | ||||
|  "bitflags", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "redox_users" | ||||
| version = "0.4.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "528532f3d801c87aec9def2add9ca802fe569e44a544afe633765267840abe64" | ||||
| dependencies = [ | ||||
|  "getrandom", | ||||
|  "redox_syscall", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "regex" | ||||
| version = "1.5.4" | ||||
| @@ -1600,15 +1502,6 @@ dependencies = [ | ||||
|  "regex-syntax", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "regex-automata" | ||||
| version = "0.1.10" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "6c230d73fb8d8c1b9c0b3135c5142a8acee3a0558fb8db5cf1cb65f8d7862132" | ||||
| dependencies = [ | ||||
|  "regex-syntax", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "regex-syntax" | ||||
| version = "0.6.25" | ||||
| @@ -1636,9 +1529,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "rtrb" | ||||
| version = "0.2.0" | ||||
| version = "0.1.4" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "7d911d82e9745a598dd118e6bda2165531e316cf8eae65240776637ae8168458" | ||||
| checksum = "318256ac02f7e11a48a10339ba5dca8bd7eb17496abf384e8ea909bb2ae5275f" | ||||
| dependencies = [ | ||||
|  "cache-padded", | ||||
| ] | ||||
| @@ -1749,9 +1642,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "serde_json" | ||||
| version = "1.0.68" | ||||
| version = "1.0.67" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "0f690853975602e1bfe1ccbf50504d67174e3bcf340f23b5ea9992e0587a52d8" | ||||
| checksum = "a7f9e390c27c3c0ce8bc5d725f6e4d30a29d26659494aa4b17535f7522c5c950" | ||||
| dependencies = [ | ||||
|  "itoa", | ||||
|  "ryu", | ||||
| @@ -1764,15 +1657,6 @@ version = "0.6.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "2579985fda508104f7587689507983eadd6a6e84dd35d6d115361f530916fa0d" | ||||
|  | ||||
| [[package]] | ||||
| name = "sharded-slab" | ||||
| version = "0.1.4" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "900fba806f70c630b0a382d0d825e17a0f19fcd059a2ade1ff237bcddf446b31" | ||||
| dependencies = [ | ||||
|  "lazy_static", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "shlex" | ||||
| version = "0.1.1" | ||||
| @@ -1801,9 +1685,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "smallvec" | ||||
| version = "1.7.0" | ||||
| version = "1.6.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "1ecab6c735a6bb4139c0caafd0cc3635748bbb3acf4550e8138122099251f309" | ||||
| checksum = "fe0f37c9e8f3c5a4a66ad655a93c74daac4ad00c441533bf5c6e7990bb42604e" | ||||
|  | ||||
| [[package]] | ||||
| name = "smithay-client-toolkit" | ||||
| @@ -1905,9 +1789,9 @@ checksum = "6446ced80d6c486436db5c078dde11a9f73d42b57fb273121e160b84f63d894c" | ||||
|  | ||||
| [[package]] | ||||
| name = "syn" | ||||
| version = "1.0.80" | ||||
| version = "1.0.76" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "d010a1623fbd906d51d650a9916aaefc05ffa0e4053ff7fe601167f3e715d194" | ||||
| checksum = "c6f107db402c2c2055242dbf4d2af0e69197202e9faacbef9571bbe47f5a1b84" | ||||
| dependencies = [ | ||||
|  "proc-macro2", | ||||
|  "quote", | ||||
| @@ -1934,38 +1818,29 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "thiserror" | ||||
| version = "1.0.30" | ||||
| version = "1.0.29" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "854babe52e4df1653706b98fcfc05843010039b406875930a70e4d9644e5c417" | ||||
| checksum = "602eca064b2d83369e2b2f34b09c70b605402801927c65c11071ac911d299b88" | ||||
| dependencies = [ | ||||
|  "thiserror-impl", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "thiserror-impl" | ||||
| version = "1.0.30" | ||||
| version = "1.0.29" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "aa32fd3f627f367fe16f893e2597ae3c05020f8bba2666a4e6ea73d377e5714b" | ||||
| checksum = "bad553cc2c78e8de258400763a647e80e6d1b31ee237275d756f6836d204494c" | ||||
| dependencies = [ | ||||
|  "proc-macro2", | ||||
|  "quote", | ||||
|  "syn", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "thread_local" | ||||
| version = "1.1.3" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "8018d24e04c95ac8790716a5987d0fec4f8b27249ffa0f7d33f1369bdfb88cbd" | ||||
| dependencies = [ | ||||
|  "once_cell", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "tinyvec" | ||||
| version = "1.5.0" | ||||
| version = "1.3.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "f83b2a3d4d9091d0abd7eba4dc2710b1718583bd4d8992e2190720ea38f391f7" | ||||
| checksum = "848a1e1181b9f6753b5e96a092749e29b11d19ede67dfbbd6c7dc7e0f49b5338" | ||||
| dependencies = [ | ||||
|  "tinyvec_macros", | ||||
| ] | ||||
| @@ -1985,81 +1860,6 @@ dependencies = [ | ||||
|  "serde", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "tracing" | ||||
| version = "0.1.29" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "375a639232caf30edfc78e8d89b2d4c375515393e7af7e16f01cd96917fb2105" | ||||
| dependencies = [ | ||||
|  "cfg-if 1.0.0", | ||||
|  "pin-project-lite", | ||||
|  "tracing-attributes", | ||||
|  "tracing-core", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "tracing-attributes" | ||||
| version = "0.1.18" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "f4f480b8f81512e825f337ad51e94c1eb5d3bbdf2b363dcd01e2b19a9ffe3f8e" | ||||
| dependencies = [ | ||||
|  "proc-macro2", | ||||
|  "quote", | ||||
|  "syn", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "tracing-core" | ||||
| version = "0.1.21" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "1f4ed65637b8390770814083d20756f87bfa2c21bf2f110babdc5438351746e4" | ||||
| dependencies = [ | ||||
|  "lazy_static", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "tracing-log" | ||||
| version = "0.1.2" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "a6923477a48e41c1951f1999ef8bb5a3023eb723ceadafe78ffb65dc366761e3" | ||||
| dependencies = [ | ||||
|  "lazy_static", | ||||
|  "log", | ||||
|  "tracing-core", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "tracing-serde" | ||||
| version = "0.1.2" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "fb65ea441fbb84f9f6748fd496cf7f63ec9af5bca94dd86456978d055e8eb28b" | ||||
| dependencies = [ | ||||
|  "serde", | ||||
|  "tracing-core", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "tracing-subscriber" | ||||
| version = "0.2.25" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "0e0d2eaa99c3c2e41547cfa109e910a68ea03823cccad4a0525dcbc9b01e8c71" | ||||
| dependencies = [ | ||||
|  "ansi_term 0.12.1", | ||||
|  "chrono", | ||||
|  "lazy_static", | ||||
|  "matchers", | ||||
|  "regex", | ||||
|  "serde", | ||||
|  "serde_json", | ||||
|  "sharded-slab", | ||||
|  "smallvec", | ||||
|  "thread_local", | ||||
|  "tracing", | ||||
|  "tracing-core", | ||||
|  "tracing-log", | ||||
|  "tracing-serde", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "ttf-parser" | ||||
| version = "0.6.2" | ||||
| @@ -2077,9 +1877,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "unicode-width" | ||||
| version = "0.1.9" | ||||
| version = "0.1.8" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "3ed742d4ea2bd1176e236172c8429aaf54486e7ac098db29ffe6529e0ce50973" | ||||
| checksum = "9337591893a19b88d8d87f2cec1e73fad5cdfd10e5a6f349f498ad6ea2ffb1e3" | ||||
|  | ||||
| [[package]] | ||||
| name = "unicode-xid" | ||||
| @@ -2116,17 +1916,11 @@ dependencies = [ | ||||
|  "winapi-util", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "wasi" | ||||
| version = "0.10.2+wasi-snapshot-preview1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "fd6fbd9a79829dd1ad0cc20627bf1ed606756a7f77edff7b66b7064f9cb327c6" | ||||
|  | ||||
| [[package]] | ||||
| name = "wasm-bindgen" | ||||
| version = "0.2.78" | ||||
| version = "0.2.77" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "632f73e236b219150ea279196e54e610f5dbafa5d61786303d4da54f84e47fce" | ||||
| checksum = "5e68338db6becec24d3c7977b5bf8a48be992c934b5d07177e3931f5dc9b076c" | ||||
| dependencies = [ | ||||
|  "cfg-if 1.0.0", | ||||
|  "wasm-bindgen-macro", | ||||
| @@ -2134,9 +1928,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "wasm-bindgen-backend" | ||||
| version = "0.2.78" | ||||
| version = "0.2.77" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "a317bf8f9fba2476b4b2c85ef4c4af8ff39c3c7f0cdfeed4f82c34a880aa837b" | ||||
| checksum = "f34c405b4f0658583dba0c1c7c9b694f3cac32655db463b56c254a1c75269523" | ||||
| dependencies = [ | ||||
|  "bumpalo", | ||||
|  "lazy_static", | ||||
| @@ -2149,9 +1943,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "wasm-bindgen-futures" | ||||
| version = "0.4.28" | ||||
| version = "0.4.27" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "8e8d7523cb1f2a4c96c1317ca690031b714a51cc14e05f712446691f413f5d39" | ||||
| checksum = "a87d738d4abc4cf22f6eb142f5b9a81301331ee3c767f2fef2fda4e325492060" | ||||
| dependencies = [ | ||||
|  "cfg-if 1.0.0", | ||||
|  "js-sys", | ||||
| @@ -2161,9 +1955,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "wasm-bindgen-macro" | ||||
| version = "0.2.78" | ||||
| version = "0.2.77" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "d56146e7c495528bf6587663bea13a8eb588d39b36b679d83972e1a2dbbdacf9" | ||||
| checksum = "b9d5a6580be83b19dc570a8f9c324251687ab2184e57086f71625feb57ec77c8" | ||||
| dependencies = [ | ||||
|  "quote", | ||||
|  "wasm-bindgen-macro-support", | ||||
| @@ -2171,9 +1965,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "wasm-bindgen-macro-support" | ||||
| version = "0.2.78" | ||||
| version = "0.2.77" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "7803e0eea25835f8abdc585cd3021b3deb11543c6fe226dcd30b228857c5c5ab" | ||||
| checksum = "e3775a030dc6f5a0afd8a84981a21cc92a781eb429acef9ecce476d0c9113e92" | ||||
| dependencies = [ | ||||
|  "proc-macro2", | ||||
|  "quote", | ||||
| @@ -2184,9 +1978,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "wasm-bindgen-shared" | ||||
| version = "0.2.78" | ||||
| version = "0.2.77" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "0237232789cf037d5480773fe568aac745bfe2afbc11a863e97901780a6b47cc" | ||||
| checksum = "c279e376c7a8e8752a8f1eaa35b7b0bee6bb9fb0cdacfa97cc3f1f289c87e2b4" | ||||
|  | ||||
| [[package]] | ||||
| name = "wayland-client" | ||||
| @@ -2197,7 +1991,7 @@ dependencies = [ | ||||
|  "bitflags", | ||||
|  "downcast-rs", | ||||
|  "libc", | ||||
|  "nix 0.20.0", | ||||
|  "nix 0.20.1", | ||||
|  "scoped-tls", | ||||
|  "wayland-commons", | ||||
|  "wayland-scanner", | ||||
| @@ -2210,7 +2004,7 @@ version = "0.28.6" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "a21817947c7011bbd0a27e11b17b337bfd022e8544b071a2641232047966fbda" | ||||
| dependencies = [ | ||||
|  "nix 0.20.0", | ||||
|  "nix 0.20.1", | ||||
|  "once_cell", | ||||
|  "smallvec", | ||||
|  "wayland-sys", | ||||
| @@ -2222,7 +2016,7 @@ version = "0.28.6" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "be610084edd1586d45e7bdd275fe345c7c1873598caa464c4fb835dee70fa65a" | ||||
| dependencies = [ | ||||
|  "nix 0.20.0", | ||||
|  "nix 0.20.1", | ||||
|  "wayland-client", | ||||
|  "xcursor", | ||||
| ] | ||||
| @@ -2263,9 +2057,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "web-sys" | ||||
| version = "0.3.55" | ||||
| version = "0.3.51" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "38eb105f1c59d9eaa6b5cdc92b859d85b926e82cb2e0945cd0c9259faa6fe9fb" | ||||
| checksum = "e828417b379f3df7111d3a2a9e5753706cae29c41f7c4029ee9fd77f3e09e582" | ||||
| dependencies = [ | ||||
|  "js-sys", | ||||
|  "wasm-bindgen", | ||||
| @@ -2273,9 +2067,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "wgpu" | ||||
| version = "0.11.0" | ||||
| version = "0.10.1" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "d1577ecc4f6992b9e965878ac594efb24eed2bdf089c11f45b3d1c5f216e2e30" | ||||
| checksum = "3d92a4fe73b1e7d7ef99938dacd49258cbf1ad87cdb5bf6efa20c27447442b45" | ||||
| dependencies = [ | ||||
|  "arrayvec", | ||||
|  "js-sys", | ||||
| @@ -2293,9 +2087,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "wgpu-core" | ||||
| version = "0.11.0" | ||||
| version = "0.10.2" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "45af76ba5545b61a6904f26cdcf4287329144ae9e12f0c23ec4c9be982d675a6" | ||||
| checksum = "425b975c319d311e051bf3afb54120a34b187f9d889edc68e347567e512774c8" | ||||
| dependencies = [ | ||||
|  "arrayvec", | ||||
|  "bitflags", | ||||
| @@ -2315,9 +2109,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "wgpu-hal" | ||||
| version = "0.11.2" | ||||
| version = "0.10.4" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "11095a81f4406b1e594dab7dc35d6508409d364e62458f2e5b07b3edc7aca517" | ||||
| checksum = "b33daf5eff68118c0aad030886655824052f82a164fd2e257211f742c6cade53" | ||||
| dependencies = [ | ||||
|  "arrayvec", | ||||
|  "ash", | ||||
| @@ -2332,30 +2126,26 @@ dependencies = [ | ||||
|  "gpu-alloc", | ||||
|  "gpu-descriptor", | ||||
|  "inplace_it", | ||||
|  "js-sys", | ||||
|  "khronos-egl", | ||||
|  "libloading 0.7.1", | ||||
|  "libloading 0.7.0", | ||||
|  "log", | ||||
|  "metal", | ||||
|  "naga", | ||||
|  "objc", | ||||
|  "parking_lot", | ||||
|  "profiling", | ||||
|  "range-alloc", | ||||
|  "raw-window-handle", | ||||
|  "renderdoc-sys", | ||||
|  "thiserror", | ||||
|  "wasm-bindgen", | ||||
|  "web-sys", | ||||
|  "wgpu-types", | ||||
|  "winapi", | ||||
| ] | ||||
|  | ||||
| [[package]] | ||||
| name = "wgpu-types" | ||||
| version = "0.11.0" | ||||
| version = "0.10.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "e15e44ba88ec415466e18e91881319e7c9e96cb905dc623305168aea65b85ccc" | ||||
| checksum = "25feb2fbf24ab3219a9f10890ceb8e1ef02b13314ed89d64a9ae99dcad883e18" | ||||
| dependencies = [ | ||||
|  "bitflags", | ||||
| ] | ||||
| @@ -2409,7 +2199,7 @@ checksum = "79610794594d5e86be473ef7763f604f2159cbac8c94debd00df8fb41e86c2f8" | ||||
| dependencies = [ | ||||
|  "bitflags", | ||||
|  "cocoa", | ||||
|  "core-foundation 0.9.2", | ||||
|  "core-foundation 0.9.1", | ||||
|  "core-graphics 0.22.2", | ||||
|  "core-video-sys", | ||||
|  "dispatch", | ||||
| @@ -2444,12 +2234,13 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "x11-dl" | ||||
| version = "2.19.1" | ||||
| version = "2.18.5" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "ea26926b4ce81a6f5d9d0f3a0bc401e5a37c6ae14a1bfaa8ff6099ca80038c59" | ||||
| checksum = "2bf981e3a5b3301209754218f962052d4d9ee97e478f4d26d4a6eced34c1fef8" | ||||
| dependencies = [ | ||||
|  "lazy_static", | ||||
|  "libc", | ||||
|  "maybe-uninit", | ||||
|  "pkg-config", | ||||
| ] | ||||
|  | ||||
| @@ -2464,12 +2255,9 @@ dependencies = [ | ||||
|  | ||||
| [[package]] | ||||
| name = "xdg" | ||||
| version = "2.4.0" | ||||
| version = "2.2.0" | ||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | ||||
| checksum = "3a23fe958c70412687039c86f578938b4a0bb50ec788e96bce4d6ab00ddd5803" | ||||
| dependencies = [ | ||||
|  "dirs", | ||||
| ] | ||||
| checksum = "d089681aa106a86fade1b0128fb5daf07d5867a509ab036d99988dec80429a57" | ||||
|  | ||||
| [[package]] | ||||
| name = "xml-rs" | ||||
|   | ||||
							
								
								
									
										21
									
								
								Cargo.toml
									
									
									
									
									
								
							
							
						
						
									
										21
									
								
								Cargo.toml
									
									
									
									
									
								
							| @@ -8,18 +8,15 @@ resolver = "2" | ||||
| # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html | ||||
|  | ||||
| [dependencies] | ||||
| anyhow = "1.0" | ||||
| bitfield = "0.13" | ||||
| clap = "2.33" | ||||
| gilrs = "0.8" | ||||
| pixels = "0.7" | ||||
| winit = "0.25" | ||||
| winit_input_helper = "0.10" | ||||
| rodio = "0.14" | ||||
| rtrb = "0.2" | ||||
| directories-next = "2.0" | ||||
| tracing = "0.1.29" | ||||
| tracing-subscriber = "0.2.25" | ||||
| anyhow = "^1.0" | ||||
| bitfield = "^0.13" | ||||
| clap = "^2.33" | ||||
| gilrs = "^0.8" | ||||
| pixels = "^0.6" | ||||
| winit = "^0.25" | ||||
| winit_input_helper = "^0.10" | ||||
| rodio = "^0.14" | ||||
| rtrb = "^0.1.4" | ||||
|  | ||||
| [profile.release] | ||||
| debug = true | ||||
|   | ||||
							
								
								
									
										40
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										40
									
								
								README.md
									
									
									
									
									
								
							| @@ -2,37 +2,13 @@ | ||||
| [](https://ci.paoda.moe/paoda/gb) | ||||
|  | ||||
| ### Status | ||||
| * From [Blargg Test ROMs](https://github.com/L-P/blargg-test-roms/) | ||||
|     * [x] cpu_instrs | ||||
|     * [x] instr_timing | ||||
|     * [x] mem_timing | ||||
|     * [x] mem_timing-2 | ||||
|     * [ ] Partially dmg_sound  | ||||
| * [x] [dmg-acid2](https://github.com/mattcurrie/dmg-acid2) | ||||
| * From [mooneye-gb](https://github.com/Gekkio/mooneye-gb): | ||||
|     * Cartridges: | ||||
|         * [x] MBC1 | ||||
|         * [ ] MBC1M | ||||
|         * [x] MBC2 | ||||
|         * [x] MBC5 | ||||
| * Implements a cycle-accurate PPU FIFO | ||||
|     * Doesn't \*exactly\* work right just yet | ||||
|  | ||||
| Supports: ROM-only, MBC1, MBC2, MBC3 and MBC5 games. | ||||
| * Passes Blargg's cpu_instrs Test | ||||
| * Renders Background & Window Tiles  | ||||
| * Implements a PPU FIFO | ||||
|  | ||||
|  | ||||
|  | ||||
| ### Controls | ||||
| Controls are defined [here](https://git.musuka.dev/paoda/gb/src/branch/main/src/joypad.rs#L114) | ||||
|  | ||||
| Key | Button | ||||
| --- | --- | ||||
| <kbd>X</kbd> | B | ||||
| <kbd>Z</kbd> | A | ||||
| <kbd>Enter</kbd> | START | ||||
| <kbd>Shift</kbd> | SELECT | ||||
|  | ||||
| Then use the Arrow keys for the D-Pad | ||||
|  | ||||
| ### Credits | ||||
| The Boot ROM found in the `bin/` directory was made by [Optix](https://github.com/Hacktix) over [here](https://github.com/Hacktix/Bootix) | ||||
| ### Notes | ||||
| * [gameboy-logs](https://github.com/wheremyfoodat/Gameboy-logs) suggests that there are still some underlying problems with the cpu implementation | ||||
| * The Sprite FIFO does not work as expected yet | ||||
| * Sound is neither emulated nor stubbed. Upon writing / reading to a APU related register the emulator will panic.  | ||||
| * Code cleanup is pending completion of some minimum viable product of the emulator | ||||
|   | ||||
										
											Binary file not shown.
										
									
								
							| @@ -1,7 +1,6 @@ | ||||
| use crate::bus::BusIo; | ||||
| use crate::emu::SM83_CLOCK_SPEED; | ||||
| use gen::SampleProducer; | ||||
| use tracing::warn; | ||||
| use types::ch1::{Sweep, SweepDirection}; | ||||
| use types::ch3::Volume as Ch3Volume; | ||||
| use types::ch4::{CounterWidth, Frequency as Ch4Frequency, PolynomialCounter}; | ||||
| @@ -55,7 +54,7 @@ impl BusIo for Apu { | ||||
|             0x26 => self.ctrl.status(self), | ||||
|             0x30..=0x3F => self.ch3.read_byte(addr), | ||||
|             _ => { | ||||
|                 warn!("Attempted read from {:#06X}", addr); | ||||
|                 eprintln!("Read 0xFF from unused IO register {:#06X} [APU]", addr); | ||||
|                 0xFF | ||||
|             } | ||||
|         } | ||||
| @@ -86,7 +85,10 @@ impl BusIo for Apu { | ||||
|             0x26 => self.set_status(byte), | ||||
|             0x30..=0x3F => self.ch3.write_byte(addr, byte), | ||||
|             _ if !self.ctrl.enabled => {} | ||||
|             _ => warn!("Attempted write of {:#04X} to {:#06X}", byte, addr), | ||||
|             _ => eprintln!( | ||||
|                 "Wrote {:#04X} to unused IO register {:#06X} [APU]", | ||||
|                 byte, addr | ||||
|             ), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -5,12 +5,33 @@ pub(crate) const SAMPLE_RATE: u32 = 48000; // Hz | ||||
| const CHANNEL_COUNT: usize = 2; | ||||
| const BUFFER_CAPACITY: usize = 2048 * CHANNEL_COUNT; // # of samples * the # of channels | ||||
|  | ||||
| pub fn init<T>() -> (SampleProducer<T>, SampleConsumer<T>) { | ||||
|     let (prod, cons) = RingBuffer::new(BUFFER_CAPACITY); | ||||
| pub struct AudioSPSC<T> { | ||||
|     inner: RingBuffer<T>, | ||||
| } | ||||
|  | ||||
| impl<T> Default for AudioSPSC<T> { | ||||
|     fn default() -> Self { | ||||
|         Self { | ||||
|             inner: RingBuffer::new(BUFFER_CAPACITY), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl<T> AudioSPSC<T> { | ||||
|     pub fn new(capacity: usize) -> Self { | ||||
|         Self { | ||||
|             inner: RingBuffer::new(capacity), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn init(self) -> (SampleProducer<T>, SampleConsumer<T>) { | ||||
|         let (prod, cons) = self.inner.split(); | ||||
|  | ||||
|         ( | ||||
|             SampleProducer { inner: prod }, | ||||
|             SampleConsumer { inner: cons }, | ||||
|         ) | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub struct SampleProducer<T> { | ||||
|   | ||||
							
								
								
									
										30
									
								
								src/bus.rs
									
									
									
									
									
								
							
							
						
						
									
										30
									
								
								src/bus.rs
									
									
									
									
									
								
							| @@ -1,5 +1,3 @@ | ||||
| use tracing::warn; | ||||
|  | ||||
| use crate::apu::Apu; | ||||
| use crate::cartridge::Cartridge; | ||||
| use crate::high_ram::HighRam; | ||||
| @@ -16,7 +14,7 @@ pub(crate) const BOOT_SIZE: usize = 0x100; | ||||
| pub struct Bus { | ||||
|     boot: Option<[u8; BOOT_SIZE]>, // Boot ROM is 256b long | ||||
|     cart: Option<Cartridge>, | ||||
|     ppu: Ppu, | ||||
|     pub(crate) ppu: Ppu, | ||||
|     work_ram: WorkRam, | ||||
|     var_ram: VariableWorkRam, | ||||
|     pub(crate) timer: Timer, | ||||
| @@ -66,12 +64,10 @@ impl Bus { | ||||
|         self.boot.is_some() | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     pub(crate) fn clock(&mut self) { | ||||
|         self.tick(4); | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     fn tick(&mut self, limit: u8) { | ||||
|         for _ in 0..limit { | ||||
|             self.timer.tick(); | ||||
| @@ -87,26 +83,6 @@ impl Bus { | ||||
|             self.oam_write_byte(dest_addr, byte); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     pub(crate) fn joyp_mut(&mut self) -> &mut Joypad { | ||||
|         &mut self.joypad | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     pub(crate) fn cart(&self) -> Option<&Cartridge> { | ||||
|         self.cart.as_ref() | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     pub(crate) fn cart_mut(&mut self) -> Option<&mut Cartridge> { | ||||
|         self.cart.as_mut() | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     pub fn ppu(&self) -> &Ppu { | ||||
|         &self.ppu | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Bus { | ||||
| @@ -258,7 +234,7 @@ impl BusIo for Bus { | ||||
|                     0x4A => self.ppu.pos.window_y, | ||||
|                     0x4B => self.ppu.pos.window_x, | ||||
|                     _ => { | ||||
|                         warn!("Attempted read from {:#06X} on IO", addr); | ||||
|                         eprintln!("Read 0xFF from unused IO register {:#06X}.", addr); | ||||
|                         0xFF | ||||
|                     } | ||||
|                 } | ||||
| @@ -375,7 +351,7 @@ impl BusIo for Bus { | ||||
|                             self.boot = None; | ||||
|                         } | ||||
|                     } | ||||
|                     _ => warn!("Attempted write of {:#04X} to {:#06X} on IO", byte, addr), | ||||
|                     _ => eprintln!("Wrote {:#04X} to unused IO register {:#06X}.", byte, addr), | ||||
|                 }; | ||||
|             } | ||||
|             0xFF80..=0xFFFE => { | ||||
|   | ||||
							
								
								
									
										165
									
								
								src/cartridge.rs
									
									
									
									
									
								
							
							
						
						
									
										165
									
								
								src/cartridge.rs
									
									
									
									
									
								
							| @@ -1,5 +1,3 @@ | ||||
| use tracing::{info, warn}; | ||||
|  | ||||
| use crate::bus::BusIo; | ||||
|  | ||||
| const RAM_SIZE_ADDRESS: usize = 0x0149; | ||||
| @@ -17,7 +15,7 @@ pub(crate) struct Cartridge { | ||||
| impl Cartridge { | ||||
|     pub(crate) fn new(memory: Vec<u8>) -> Self { | ||||
|         let title = Self::find_title(&memory); | ||||
|         info!("Title: {:?}", title); | ||||
|         eprintln!("Cartridge Title: {:?}", title); | ||||
|  | ||||
|         Self { | ||||
|             mbc: Self::detect_mbc(&memory), | ||||
| @@ -26,14 +24,6 @@ impl Cartridge { | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub(crate) fn ext_ram(&self) -> Option<&[u8]> { | ||||
|         self.mbc.ext_ram() | ||||
|     } | ||||
|  | ||||
|     pub(crate) fn write_ext_ram(&mut self, memory: Vec<u8>) { | ||||
|         self.mbc.write_ext_ram(memory) | ||||
|     } | ||||
|  | ||||
|     fn detect_mbc(memory: &[u8]) -> Box<dyn MBCIo> { | ||||
|         let ram_size = Self::detect_ram_info(memory); | ||||
|         let rom_size = Self::detect_rom_info(memory); | ||||
| @@ -41,20 +31,20 @@ impl Cartridge { | ||||
|         let ram_cap = ram_size.capacity(); | ||||
|         let rom_cap = rom_size.capacity(); | ||||
|  | ||||
|         info!("RAM size: {} bytes", ram_cap); | ||||
|         info!("ROM size: {} bytes", rom_size.capacity()); | ||||
|         info!("MBC kind: {:?}", mbc_kind); | ||||
|         eprintln!("Cartridge Ram Size: {} bytes", ram_cap); | ||||
|         eprintln!("Cartridge ROM Size: {} bytes", rom_size.capacity()); | ||||
|         eprintln!("MBC Type: {:?}", mbc_kind); | ||||
|  | ||||
|         match mbc_kind { | ||||
|             MBCKind::None => Box::new(NoMBC), | ||||
|             MBCKind::MBC1 => Box::new(MBC1::new(ram_size, rom_size)), | ||||
|             MBCKind::MBC1WithBattery => Box::new(MBC1::with_battery(ram_size, rom_size)), | ||||
|             MBCKind::MBC1WithBattery => Box::new(MBC1::new(ram_size, rom_size)), // TODO: Implement Saving | ||||
|             MBCKind::MBC2 => Box::new(MBC2::new(rom_cap)), | ||||
|             MBCKind::MBC2WithBattery => Box::new(MBC2::with_battery(rom_cap)), | ||||
|             MBCKind::MBC2WithBattery => Box::new(MBC2::new(rom_cap)), // TODO: Implement Saving | ||||
|             MBCKind::MBC3 => Box::new(MBC3::new(ram_cap)), | ||||
|             MBCKind::MBC3WithBattery => Box::new(MBC3::with_battery(ram_cap)), | ||||
|             MBCKind::MBC3WithBattery => Box::new(MBC3::new(ram_cap)), // TODO: Implement Saving | ||||
|             MBCKind::MBC5 => Box::new(MBC5::new(ram_cap, rom_cap)), | ||||
|             MBCKind::MBC5WithBattery => Box::new(MBC5::with_battery(ram_cap, rom_cap)), | ||||
|             MBCKind::MBC5WithBattery => Box::new(MBC5::new(ram_cap, rom_cap)), // TDO: Implement Saving | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -127,8 +117,6 @@ struct MBC1 { | ||||
|     memory: Vec<u8>, | ||||
|     rom_size: RomSize, | ||||
|     mem_enabled: bool, | ||||
|  | ||||
|     has_battery: bool, | ||||
| } | ||||
|  | ||||
| impl MBC1 { | ||||
| @@ -141,20 +129,6 @@ impl MBC1 { | ||||
|             ram_bank: Default::default(), | ||||
|             mode: Default::default(), | ||||
|             mem_enabled: Default::default(), | ||||
|             has_battery: Default::default(), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn with_battery(ram_size: RamSize, rom_size: RomSize) -> Self { | ||||
|         Self { | ||||
|             rom_bank: 0x01, | ||||
|             memory: vec![0; ram_size.capacity() as usize], | ||||
|             ram_size, | ||||
|             rom_size, | ||||
|             ram_bank: Default::default(), | ||||
|             mode: Default::default(), | ||||
|             mem_enabled: Default::default(), | ||||
|             has_battery: true, | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -223,21 +197,6 @@ impl MBC1 { | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Savable for MBC1 { | ||||
|     fn ext_ram(&self) -> Option<&[u8]> { | ||||
|         match self.has_battery { | ||||
|             true => Some(&self.memory), | ||||
|             false => None, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn write_ext_ram(&mut self, memory: Vec<u8>) { | ||||
|         if self.has_battery { | ||||
|             self.memory.copy_from_slice(&memory); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl MBCIo for MBC1 { | ||||
|     fn handle_read(&self, addr: u16) -> MBCResult { | ||||
|         use MBCResult::*; | ||||
| @@ -297,8 +256,6 @@ struct MBC3 { | ||||
|  | ||||
|     // RTC Data Latch Previous Write | ||||
|     prev_latch_write: Option<u8>, | ||||
|  | ||||
|     has_battery: bool, | ||||
| } | ||||
|  | ||||
| impl MBC3 { | ||||
| @@ -310,34 +267,6 @@ impl MBC3 { | ||||
|             devs_enabled: Default::default(), | ||||
|             mapped: Default::default(), | ||||
|             prev_latch_write: Default::default(), | ||||
|             has_battery: Default::default(), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn with_battery(ram_cap: usize) -> Self { | ||||
|         Self { | ||||
|             memory: vec![0; ram_cap], | ||||
|             rom_bank: Default::default(), | ||||
|             ram_bank: Default::default(), | ||||
|             devs_enabled: Default::default(), | ||||
|             mapped: Default::default(), | ||||
|             prev_latch_write: Default::default(), | ||||
|             has_battery: true, | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Savable for MBC3 { | ||||
|     fn ext_ram(&self) -> Option<&[u8]> { | ||||
|         match self.has_battery { | ||||
|             true => Some(&self.memory), | ||||
|             false => None, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn write_ext_ram(&mut self, memory: Vec<u8>) { | ||||
|         if self.has_battery { | ||||
|             self.memory.copy_from_slice(&memory); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -413,10 +342,9 @@ struct MBC5 { | ||||
|     ram_bank: u8, | ||||
|  | ||||
|     rom_cap: usize, | ||||
|  | ||||
|     memory: Vec<u8>, | ||||
|     mem_enabled: bool, | ||||
|  | ||||
|     has_battery: bool, | ||||
| } | ||||
|  | ||||
| impl MBC5 { | ||||
| @@ -427,18 +355,6 @@ impl MBC5 { | ||||
|             rom_cap, | ||||
|             ram_bank: Default::default(), | ||||
|             mem_enabled: Default::default(), | ||||
|             has_battery: Default::default(), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn with_battery(ram_cap: usize, rom_cap: usize) -> Self { | ||||
|         Self { | ||||
|             rom_bank: 0x01, | ||||
|             memory: vec![0; ram_cap], | ||||
|             rom_cap, | ||||
|             ram_bank: Default::default(), | ||||
|             mem_enabled: Default::default(), | ||||
|             has_battery: true, | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -447,21 +363,6 @@ impl MBC5 { | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Savable for MBC5 { | ||||
|     fn ext_ram(&self) -> Option<&[u8]> { | ||||
|         match self.has_battery { | ||||
|             true => Some(&self.memory), | ||||
|             false => None, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn write_ext_ram(&mut self, memory: Vec<u8>) { | ||||
|         if self.has_battery { | ||||
|             self.memory.copy_from_slice(&memory); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl MBCIo for MBC5 { | ||||
|     fn handle_read(&self, addr: u16) -> MBCResult { | ||||
|         use MBCResult::*; | ||||
| @@ -500,7 +401,6 @@ struct MBC2 { | ||||
|     mem_enabled: bool, | ||||
|  | ||||
|     rom_cap: usize, | ||||
|     has_battery: bool, | ||||
| } | ||||
|  | ||||
| impl MBC2 { | ||||
| @@ -510,19 +410,8 @@ impl MBC2 { | ||||
|         Self { | ||||
|             rom_bank: 0x01, | ||||
|             memory: Box::new([0; Self::RAM_SIZE]), | ||||
|             rom_cap, | ||||
|             mem_enabled: Default::default(), | ||||
|             has_battery: Default::default(), | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn with_battery(rom_cap: usize) -> Self { | ||||
|         Self { | ||||
|             rom_bank: 0x01, | ||||
|             memory: Box::new([0; Self::RAM_SIZE]), | ||||
|             rom_cap, | ||||
|             mem_enabled: Default::default(), | ||||
|             has_battery: true, | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -531,21 +420,6 @@ impl MBC2 { | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Savable for MBC2 { | ||||
|     fn ext_ram(&self) -> Option<&[u8]> { | ||||
|         match self.has_battery { | ||||
|             true => Some(self.memory.as_ref()), | ||||
|             false => None, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn write_ext_ram(&mut self, memory: Vec<u8>) { | ||||
|         if self.has_battery { | ||||
|             self.memory.copy_from_slice(&memory); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl MBCIo for MBC2 { | ||||
|     fn handle_read(&self, addr: u16) -> MBCResult { | ||||
|         use MBCResult::*; | ||||
| @@ -583,27 +457,17 @@ impl MBCIo for MBC2 { | ||||
| #[derive(Debug)] | ||||
| struct NoMBC; | ||||
|  | ||||
| impl Savable for NoMBC { | ||||
|     fn ext_ram(&self) -> Option<&[u8]> { | ||||
|         None | ||||
|     } | ||||
|  | ||||
|     fn write_ext_ram(&mut self, _memory: Vec<u8>) { | ||||
|         // Nothing Happens Here | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl MBCIo for NoMBC { | ||||
|     fn handle_read(&self, addr: u16) -> MBCResult { | ||||
|         MBCResult::Address(addr as usize) | ||||
|     } | ||||
|  | ||||
|     fn handle_write(&mut self, _: u16, byte: u8) { | ||||
|         warn!("Attempted write of {:#04X} to cartridge w/out MBC", byte); | ||||
|     fn handle_write(&mut self, _addr: u16, _byte: u8) { | ||||
|         // eprintln!("Tried to write {:#04X} to a read-only cartridge", byte); | ||||
|     } | ||||
| } | ||||
|  | ||||
| trait MBCIo: Savable { | ||||
| trait MBCIo { | ||||
|     fn handle_read(&self, addr: u16) -> MBCResult; | ||||
|     fn handle_write(&mut self, addr: u16, byte: u8); | ||||
| } | ||||
| @@ -743,8 +607,3 @@ impl Default for Box<dyn MBCIo> { | ||||
|         Box::new(NoMBC) | ||||
|     } | ||||
| } | ||||
|  | ||||
| trait Savable { | ||||
|     fn ext_ram(&self) -> Option<&[u8]>; | ||||
|     fn write_ext_ram(&mut self, memory: Vec<u8>); | ||||
| } | ||||
|   | ||||
							
								
								
									
										25
									
								
								src/cpu.rs
									
									
									
									
									
								
							
							
						
						
									
										25
									
								
								src/cpu.rs
									
									
									
									
									
								
							| @@ -7,7 +7,7 @@ use std::fmt::{Display, Formatter, Result as FmtResult}; | ||||
|  | ||||
| #[derive(Debug, Default)] | ||||
| pub struct Cpu { | ||||
|     pub(crate) bus: Bus, | ||||
|     pub bus: Bus, | ||||
|     reg: Registers, | ||||
|     flags: Flags, | ||||
|     ime: ImeState, | ||||
| @@ -15,7 +15,6 @@ pub struct Cpu { | ||||
| } | ||||
|  | ||||
| impl Cpu { | ||||
|     #[allow(dead_code)] | ||||
|     pub(crate) fn without_boot() -> Self { | ||||
|         Self { | ||||
|             reg: Registers { | ||||
| @@ -58,7 +57,10 @@ impl Cpu { | ||||
|     } | ||||
|  | ||||
|     pub(crate) fn is_halted(&self) -> bool { | ||||
|         matches!(self.state, State::Halt(_)) | ||||
|         match self.state { | ||||
|             State::Halt(_) => true, | ||||
|             _ => false, | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub(crate) fn halt_kind(&self) -> Option<HaltKind> { | ||||
| @@ -118,10 +120,13 @@ impl Cpu { | ||||
|             use HaltKind::*; | ||||
|  | ||||
|             self.bus.clock(); | ||||
|             return match kind { | ||||
|  | ||||
|             let elapsed = match kind { | ||||
|                 ImeEnabled | NonePending => 4, | ||||
|                 SomePending => todo!("Implement HALT bug"), | ||||
|             }; | ||||
|  | ||||
|             return elapsed; | ||||
|         } | ||||
|  | ||||
|         let opcode = self.fetch(); | ||||
| @@ -130,11 +135,11 @@ impl Cpu { | ||||
|         self.handle_ei(); | ||||
|  | ||||
|         // For use in Blargg's Test ROMs | ||||
|         // if self.read_byte(0xFF02) == 0x81 { | ||||
|         //     let c = self.read_byte(0xFF01) as char; | ||||
|         //     self.write_byte(0xFF02, 0x00); | ||||
|         //     eprint!("{}", c); | ||||
|         // } | ||||
|         if self.read_byte(0xFF02) == 0x81 { | ||||
|             let c = self.read_byte(0xFF01) as char; | ||||
|             self.write_byte(0xFF02, 0x00); | ||||
|             eprint!("{}", c); | ||||
|         } | ||||
|  | ||||
|         elapsed | ||||
|     } | ||||
| @@ -151,12 +156,10 @@ impl BusIo for Cpu { | ||||
| } | ||||
|  | ||||
| impl Cpu { | ||||
|     #[inline] | ||||
|     pub(crate) fn bus(&self) -> &Bus { | ||||
|         &self.bus | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     pub(crate) fn bus_mut(&mut self) -> &mut Bus { | ||||
|         &mut self.bus | ||||
|     } | ||||
|   | ||||
							
								
								
									
										74
									
								
								src/emu.rs
									
									
									
									
									
								
							
							
						
						
									
										74
									
								
								src/emu.rs
									
									
									
									
									
								
							| @@ -2,11 +2,7 @@ use crate::apu::gen::SampleProducer; | ||||
| use crate::cpu::Cpu; | ||||
| use crate::joypad::{self, Joypad}; | ||||
| use crate::{Cycle, GB_HEIGHT, GB_WIDTH}; | ||||
| use clap::crate_name; | ||||
| use gilrs::Gilrs; | ||||
| use std::fs::File; | ||||
| use std::io::{Read, Write}; | ||||
| use std::path::PathBuf; | ||||
| use std::time::Duration; | ||||
| use winit_input_helper::WinitInputHelper; | ||||
|  | ||||
| @@ -21,8 +17,8 @@ pub fn run_frame(emu: &mut Emulator, gamepad: &mut Gilrs, key: &WinitInputHelper | ||||
|     if let Some(event) = gamepad.next_event() { | ||||
|         joypad::handle_gamepad_input(emu.joyp_mut(), event); | ||||
|     } | ||||
|     joypad::handle_keyboard_input(emu.joyp_mut(), key); | ||||
|  | ||||
|     joypad::handle_keyboard_input(emu.joyp_mut(), key); | ||||
|     while elapsed < CYCLES_IN_FRAME { | ||||
|         elapsed += emu.step(); | ||||
|     } | ||||
| @@ -31,7 +27,7 @@ pub fn run_frame(emu: &mut Emulator, gamepad: &mut Gilrs, key: &WinitInputHelper | ||||
| } | ||||
|  | ||||
| pub fn draw_frame(emu: &Emulator, buf: &mut [u8; GB_HEIGHT * GB_WIDTH * 4]) { | ||||
|     buf.copy_from_slice(emu.cpu.bus().ppu().frame_buf()); | ||||
|     buf.copy_from_slice(emu.cpu.bus().ppu.frame_buf()); | ||||
| } | ||||
|  | ||||
| pub struct Emulator { | ||||
| @@ -48,18 +44,15 @@ impl Emulator { | ||||
|     } | ||||
|  | ||||
|     fn step(&mut self) -> Cycle { | ||||
|         let cycles = self.cpu.step(); | ||||
|         self.timestamp += cycles; | ||||
|         cycles | ||||
|         self.cpu.step() | ||||
|     } | ||||
|  | ||||
|     fn load_cart(&mut self, rom: Vec<u8>) { | ||||
|         self.cpu.bus_mut().load_cart(rom); | ||||
|         self.cpu.bus_mut().load_cart(rom) | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     fn joyp_mut(&mut self) -> &mut Joypad { | ||||
|         self.cpu.bus_mut().joyp_mut() | ||||
|         &mut self.cpu.bus_mut().joypad | ||||
|     } | ||||
|  | ||||
|     pub fn set_prod(&mut self, prod: SampleProducer<f32>) { | ||||
| @@ -69,51 +62,6 @@ impl Emulator { | ||||
|     pub fn title(&self) -> &str { | ||||
|         self.cpu.bus().cart_title().unwrap_or(DEFAULT_TITLE) | ||||
|     } | ||||
|  | ||||
|     pub fn try_write_sav(&self) -> std::io::Result<()> { | ||||
|         if let Some(ext_ram) = self.cpu.bus().cart().map(|c| c.ext_ram()).flatten() { | ||||
|             if let Some(title) = self.cpu.bus().cart_title() { | ||||
|                 let mut save_path = Self::data_path().unwrap_or_else(|| PathBuf::from(".")); | ||||
|                 save_path.push(title); | ||||
|                 save_path.set_extension("sav"); | ||||
|  | ||||
|                 let mut file = File::create(save_path)?; | ||||
|                 file.write_all(ext_ram)?; | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     pub fn try_load_sav(&mut self) -> std::io::Result<()> { | ||||
|         if let Some(cart) = self.cpu.bus_mut().cart_mut() { | ||||
|             if let Some(title) = cart.title() { | ||||
|                 let mut save_path = Self::data_path().unwrap_or_else(|| PathBuf::from(".")); | ||||
|                 save_path.push(title); | ||||
|                 save_path.set_extension("sav"); | ||||
|  | ||||
|                 if let Ok(mut file) = File::open(save_path) { | ||||
|                     let mut memory = Vec::new(); | ||||
|                     file.read_to_end(&mut memory)?; | ||||
|  | ||||
|                     cart.write_ext_ram(memory); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         Ok(()) | ||||
|     } | ||||
|  | ||||
|     fn data_path() -> Option<PathBuf> { | ||||
|         match directories_next::ProjectDirs::from("dev", "musuka", crate_name!()) { | ||||
|             Some(dirs) => { | ||||
|                 let data_local = dirs.data_local_dir(); | ||||
|                 std::fs::create_dir_all(data_local).ok()?; | ||||
|                 Some(data_local.to_path_buf()) | ||||
|             } | ||||
|             None => None, | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| pub mod build { | ||||
| @@ -121,8 +69,6 @@ pub mod build { | ||||
|     use std::io::{Read, Result}; | ||||
|     use std::path::Path; | ||||
|  | ||||
|     use tracing::info; | ||||
|  | ||||
|     use crate::bus::BOOT_SIZE; | ||||
|     use crate::cpu::Cpu; | ||||
|  | ||||
| @@ -161,14 +107,8 @@ pub mod build { | ||||
|  | ||||
|         pub fn finish(mut self) -> Emulator { | ||||
|             let mut emu = Emulator::new(match self.boot { | ||||
|                 Some(rom) => { | ||||
|                     info!("User-provided Boot ROM"); | ||||
|                     Cpu::with_boot(rom) | ||||
|                 } | ||||
|                 None => { | ||||
|                     info!("Built-in Boot ROM"); | ||||
|                     Cpu::with_boot(*include_bytes!("../bin/bootix_dmg.bin")) | ||||
|                 } | ||||
|                 Some(rom) => Cpu::with_boot(rom), | ||||
|                 None => Cpu::without_boot(), | ||||
|             }); | ||||
|  | ||||
|             if let Some(rom) = self.cart.take() { | ||||
|   | ||||
| @@ -116,7 +116,7 @@ impl std::fmt::Debug for Instruction { | ||||
| impl Instruction { | ||||
|     pub(crate) fn execute(cpu: &mut Cpu, instruction: Self) -> Cycle { | ||||
|         match instruction { | ||||
|             Instruction::NOP => 4, | ||||
|             Instruction::NOP => (4), | ||||
|             Instruction::LD(target, src) => match (target, src) { | ||||
|                 (LDTarget::IndirectImmediateWord, LDSource::SP) => { | ||||
|                     // LD (u16), SP | Store stack pointer in byte at 16-bit register | ||||
| @@ -361,12 +361,12 @@ impl Instruction { | ||||
|                     let (cycles, sum) = match reg { | ||||
|                         B | C | D | E | H | L | A => { | ||||
|                             let right = cpu.register(reg.cpu_register()); | ||||
|                             (4, Self::add(left, right, &mut flags)) | ||||
|                             ((4), Self::add(left, right, &mut flags)) | ||||
|                         } | ||||
|                         IndirectHL => { | ||||
|                             let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                             let right = Self::read_byte(&mut cpu.bus, addr); | ||||
|                             (8, Self::add(left, right, &mut flags)) | ||||
|                             ((8), Self::add(left, right, &mut flags)) | ||||
|                         } | ||||
|                     }; | ||||
|  | ||||
| @@ -610,13 +610,13 @@ impl Instruction { | ||||
|                         B | C | D | E | H | L | A => { | ||||
|                             let right = cpu.register(reg.cpu_register()); | ||||
|                             let sum = Self::add_with_carry_bit(left, right, flags.c(), &mut flags); | ||||
|                             (4, sum) | ||||
|                             ((4), sum) | ||||
|                         } | ||||
|                         IndirectHL => { | ||||
|                             let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                             let right = Self::read_byte(&mut cpu.bus, addr); | ||||
|                             let sum = Self::add_with_carry_bit(left, right, flags.c(), &mut flags); | ||||
|                             (8, sum) | ||||
|                             ((8), sum) | ||||
|                         } | ||||
|                     }; | ||||
|                     cpu.set_register(CpuRegister::A, sum); | ||||
| @@ -646,12 +646,12 @@ impl Instruction { | ||||
|                     let (cycles, diff) = match reg { | ||||
|                         B | C | D | E | H | L | A => { | ||||
|                             let right = cpu.register(reg.cpu_register()); | ||||
|                             (4, Self::sub(left, right, &mut flags)) | ||||
|                             ((4), Self::sub(left, right, &mut flags)) | ||||
|                         } | ||||
|                         IndirectHL => { | ||||
|                             let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                             let right = Self::read_byte(&mut cpu.bus, addr); | ||||
|                             (8, Self::sub(left, right, &mut flags)) | ||||
|                             ((8), Self::sub(left, right, &mut flags)) | ||||
|                         } | ||||
|                     }; | ||||
|                     cpu.set_register(CpuRegister::A, diff); | ||||
| @@ -681,13 +681,13 @@ impl Instruction { | ||||
|                         B | C | D | E | H | L | A => { | ||||
|                             let right = cpu.register(reg.cpu_register()); | ||||
|                             let diff = Self::sub_with_carry(left, right, flags.c(), &mut flags); | ||||
|                             (4, diff) | ||||
|                             ((4), diff) | ||||
|                         } | ||||
|                         IndirectHL => { | ||||
|                             let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                             let right = Self::read_byte(&mut cpu.bus, addr); | ||||
|                             let diff = Self::sub_with_carry(left, right, flags.c(), &mut flags); | ||||
|                             (8, diff) | ||||
|                             ((8), diff) | ||||
|                         } | ||||
|                     }; | ||||
|                     cpu.set_register(CpuRegister::A, diff); | ||||
| @@ -717,7 +717,7 @@ impl Instruction { | ||||
|                         IndirectHL => { | ||||
|                             let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                             let right = Self::read_byte(&mut cpu.bus, addr); | ||||
|                             (8, left & right) | ||||
|                             ((8), left & right) | ||||
|                         } | ||||
|                     }; | ||||
|                     cpu.set_register(CpuRegister::A, acc); | ||||
| @@ -743,7 +743,7 @@ impl Instruction { | ||||
|                         IndirectHL => { | ||||
|                             let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                             let right = Self::read_byte(&mut cpu.bus, addr); | ||||
|                             (8, left ^ right) | ||||
|                             ((8), left ^ right) | ||||
|                         } | ||||
|                     }; | ||||
|                     cpu.set_register(CpuRegister::A, acc); | ||||
| @@ -769,7 +769,7 @@ impl Instruction { | ||||
|                         IndirectHL => { | ||||
|                             let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                             let right = Self::read_byte(&mut cpu.bus, addr); | ||||
|                             (8, left | right) | ||||
|                             ((8), left | right) | ||||
|                         } | ||||
|                     }; | ||||
|                     cpu.set_register(CpuRegister::A, acc); | ||||
| @@ -1053,14 +1053,14 @@ impl Instruction { | ||||
|                         let byte = cpu.register(reg); | ||||
|                         let rotated = byte.rotate_left(1); | ||||
|                         cpu.set_register(reg, rotated); | ||||
|                         (8, byte >> 7, rotated) | ||||
|                         ((8), byte >> 7, rotated) | ||||
|                     } | ||||
|                     IndirectHL => { | ||||
|                         let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                         let byte = Self::read_byte(&mut cpu.bus, addr); | ||||
|                         let rotated = byte.rotate_left(1); | ||||
|                         Self::write_byte(&mut cpu.bus, addr, rotated); | ||||
|                         (16, byte >> 7, rotated) | ||||
|                         ((16), byte >> 7, rotated) | ||||
|                     } | ||||
|                 }; | ||||
|                 cpu.update_flags(rotated == 0, false, false, most_sgfnt == 0x01); | ||||
| @@ -1076,14 +1076,14 @@ impl Instruction { | ||||
|                         let byte = cpu.register(reg); | ||||
|                         let rotated = byte.rotate_right(1); | ||||
|                         cpu.set_register(reg, rotated); | ||||
|                         (8, byte & 0x01, rotated) | ||||
|                         ((8), byte & 0x01, rotated) | ||||
|                     } | ||||
|                     IndirectHL => { | ||||
|                         let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                         let byte = Self::read_byte(&mut cpu.bus, addr); | ||||
|                         let rotated = byte.rotate_right(1); | ||||
|                         Self::write_byte(&mut cpu.bus, addr, rotated); | ||||
|                         (16, byte & 0x01, rotated) | ||||
|                         ((16), byte & 0x01, rotated) | ||||
|                     } | ||||
|                 }; | ||||
|                 cpu.update_flags(rotated == 0, false, false, least_sgfnt == 0x01); | ||||
| @@ -1101,14 +1101,14 @@ impl Instruction { | ||||
|                         let byte = cpu.register(reg); | ||||
|                         let (rotated, carry) = Self::rl_thru_carry(byte, flags.c()); | ||||
|                         cpu.set_register(reg, rotated); | ||||
|                         (8, rotated, carry) | ||||
|                         ((8), rotated, carry) | ||||
|                     } | ||||
|                     IndirectHL => { | ||||
|                         let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                         let byte = Self::read_byte(&mut cpu.bus, addr); | ||||
|                         let (rotated, carry) = Self::rl_thru_carry(byte, flags.c()); | ||||
|                         Self::write_byte(&mut cpu.bus, addr, rotated); | ||||
|                         (16, rotated, carry) | ||||
|                         ((16), rotated, carry) | ||||
|                     } | ||||
|                 }; | ||||
|                 cpu.update_flags(rotated == 0, false, false, carry); | ||||
| @@ -1126,14 +1126,14 @@ impl Instruction { | ||||
|                         let byte = cpu.register(reg); | ||||
|                         let (rotated, carry) = Self::rr_thru_carry(byte, flags.c()); | ||||
|                         cpu.set_register(reg, rotated); | ||||
|                         (8, rotated, carry) | ||||
|                         ((8), rotated, carry) | ||||
|                     } | ||||
|                     IndirectHL => { | ||||
|                         let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                         let byte = Self::read_byte(&mut cpu.bus, addr); | ||||
|                         let (rotated, carry) = Self::rr_thru_carry(byte, flags.c()); | ||||
|                         Self::write_byte(&mut cpu.bus, addr, rotated); | ||||
|                         (16, rotated, carry) | ||||
|                         ((16), rotated, carry) | ||||
|                     } | ||||
|                 }; | ||||
|                 cpu.update_flags(rotated == 0, false, false, carry); | ||||
| @@ -1149,14 +1149,14 @@ impl Instruction { | ||||
|                         let byte = cpu.register(reg); | ||||
|                         let shifted = byte << 1; | ||||
|                         cpu.set_register(reg, shifted); | ||||
|                         (8, (byte >> 7) & 0x01, shifted) | ||||
|                         ((8), (byte >> 7) & 0x01, shifted) | ||||
|                     } | ||||
|                     IndirectHL => { | ||||
|                         let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                         let byte = Self::read_byte(&mut cpu.bus, addr); | ||||
|                         let shifted = byte << 1; | ||||
|                         Self::write_byte(&mut cpu.bus, addr, shifted); | ||||
|                         (16, (byte >> 7) & 0x01, shifted) | ||||
|                         ((16), (byte >> 7) & 0x01, shifted) | ||||
|                     } | ||||
|                 }; | ||||
|                 cpu.update_flags(shifted == 0, false, false, most_sgfnt == 0x01); | ||||
| @@ -1172,14 +1172,14 @@ impl Instruction { | ||||
|                         let byte = cpu.register(reg); | ||||
|                         let shifted = ((byte >> 7) & 0x01) << 7 | byte >> 1; | ||||
|                         cpu.set_register(reg, shifted); | ||||
|                         (8, byte & 0x01, shifted) | ||||
|                         ((8), byte & 0x01, shifted) | ||||
|                     } | ||||
|                     IndirectHL => { | ||||
|                         let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                         let byte = Self::read_byte(&mut cpu.bus, addr); | ||||
|                         let shifted = ((byte >> 7) & 0x01) << 7 | byte >> 1; | ||||
|                         Self::write_byte(&mut cpu.bus, addr, shifted); | ||||
|                         (16, byte & 0x01, shifted) | ||||
|                         ((16), byte & 0x01, shifted) | ||||
|                     } | ||||
|                 }; | ||||
|                 cpu.update_flags(shifted == 0, false, false, least_sgfnt == 0x01); | ||||
| @@ -1194,14 +1194,14 @@ impl Instruction { | ||||
|                         let reg = reg.cpu_register(); | ||||
|                         let swapped = Self::swap_bits(cpu.register(reg)); | ||||
|                         cpu.set_register(reg, swapped); | ||||
|                         (8, swapped) | ||||
|                         ((8), swapped) | ||||
|                     } | ||||
|  | ||||
|                     IndirectHL => { | ||||
|                         let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                         let swapped = Self::swap_bits(Self::read_byte(&mut cpu.bus, addr)); | ||||
|                         Self::write_byte(&mut cpu.bus, addr, swapped); | ||||
|                         (16, swapped) | ||||
|                         ((16), swapped) | ||||
|                     } | ||||
|                 }; | ||||
|                 cpu.update_flags(swapped == 0, false, false, false); | ||||
| @@ -1217,14 +1217,14 @@ impl Instruction { | ||||
|                         let byte = cpu.register(reg); | ||||
|                         let shifted = byte >> 1; | ||||
|                         cpu.set_register(reg, shifted); | ||||
|                         (8, byte & 0x01, shifted) | ||||
|                         ((8), byte & 0x01, shifted) | ||||
|                     } | ||||
|                     IndirectHL => { | ||||
|                         let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                         let byte = Self::read_byte(&mut cpu.bus, addr); | ||||
|                         let shifted = byte >> 1; | ||||
|                         Self::write_byte(&mut cpu.bus, addr, shifted); | ||||
|                         (16, byte & 0x01, shifted) | ||||
|                         ((16), byte & 0x01, shifted) | ||||
|                     } | ||||
|                 }; | ||||
|                 cpu.update_flags(shift_reg == 0, false, false, least_sgfnt == 0x01); | ||||
| @@ -1239,12 +1239,12 @@ impl Instruction { | ||||
|                     B | C | D | E | H | L | A => { | ||||
|                         let reg = reg.cpu_register(); | ||||
|                         let byte = cpu.register(reg); | ||||
|                         (8, ((byte >> bit) & 0x01) == 0x01) | ||||
|                         ((8), ((byte >> bit) & 0x01) == 0x01) | ||||
|                     } | ||||
|                     IndirectHL => { | ||||
|                         let addr = cpu.register_pair(RegisterPair::HL); | ||||
|                         let byte = Self::read_byte(&mut cpu.bus, addr); | ||||
|                         (12, ((byte >> bit) & 0x01) == 0x01) | ||||
|                         ((12), ((byte >> bit) & 0x01) == 0x01) | ||||
|                     } | ||||
|                 }; | ||||
|                 flags.set_z(!is_set); | ||||
|   | ||||
| @@ -110,7 +110,6 @@ impl ButtonEvent { | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[inline] | ||||
| pub fn handle_keyboard_input(pad: &mut Joypad, input: &WinitInputHelper) { | ||||
|     use winit::event::VirtualKeyCode; | ||||
|  | ||||
| @@ -147,17 +146,17 @@ pub fn handle_keyboard_input(pad: &mut Joypad, input: &WinitInputHelper) { | ||||
|         state.dpad_right.update(false, irq); | ||||
|     } | ||||
|  | ||||
|     if input.key_pressed(VirtualKeyCode::Return) { | ||||
|     if input.key_pressed(VirtualKeyCode::T) { | ||||
|         state.start.update(true, irq); | ||||
|     } | ||||
|     if input.key_released(VirtualKeyCode::Return) { | ||||
|     if input.key_released(VirtualKeyCode::T) { | ||||
|         state.start.update(false, irq); | ||||
|     } | ||||
|  | ||||
|     if input.key_pressed(VirtualKeyCode::RShift) { | ||||
|     if input.key_pressed(VirtualKeyCode::Y) { | ||||
|         state.select.update(true, irq); | ||||
|     } | ||||
|     if input.key_released(VirtualKeyCode::RShift) { | ||||
|     if input.key_released(VirtualKeyCode::Y) { | ||||
|         state.select.update(false, irq); | ||||
|     } | ||||
|  | ||||
| @@ -176,7 +175,6 @@ pub fn handle_keyboard_input(pad: &mut Joypad, input: &WinitInputHelper) { | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[inline] | ||||
| pub fn handle_gamepad_input(pad: &mut Joypad, event: GamepadEvent) { | ||||
|     use Button::*; | ||||
|     use GamepadEventType::*; | ||||
|   | ||||
| @@ -1,4 +1,4 @@ | ||||
| pub use apu::gen::init as spsc_init; | ||||
| pub use apu::gen::AudioSPSC; | ||||
| pub type Cycle = u64; | ||||
|  | ||||
| pub const GB_WIDTH: usize = 160; | ||||
|   | ||||
							
								
								
									
										41
									
								
								src/main.rs
									
									
									
									
									
								
							
							
						
						
									
										41
									
								
								src/main.rs
									
									
									
									
									
								
							| @@ -1,15 +1,12 @@ | ||||
| use std::convert::TryInto; | ||||
|  | ||||
| use anyhow::Result; | ||||
| use anyhow::{anyhow, Result}; | ||||
| use clap::{crate_authors, crate_description, crate_name, crate_version, App, Arg}; | ||||
| use gb::emu::build::EmulatorBuilder; | ||||
| use gb::emu::CYCLES_IN_FRAME; | ||||
| use gb::{Cycle, GB_HEIGHT, GB_WIDTH}; | ||||
| use gb::{AudioSPSC, Cycle, GB_HEIGHT, GB_WIDTH}; | ||||
| use gilrs::Gilrs; | ||||
| use pixels::{PixelsBuilder, SurfaceTexture}; | ||||
| use rodio::{OutputStream, Sink}; | ||||
| use tracing::info; | ||||
| use tracing_subscriber::EnvFilter; | ||||
| use winit::dpi::{LogicalSize, PhysicalSize}; | ||||
| use winit::event::{Event, VirtualKeyCode}; | ||||
| use winit::event_loop::{ControlFlow, EventLoop}; | ||||
| @@ -17,7 +14,7 @@ use winit::window::{Window, WindowBuilder}; | ||||
| use winit_input_helper::WinitInputHelper; | ||||
|  | ||||
| const WINDOW_SCALE: usize = 3; | ||||
| const AUDIO_ENABLED: bool = true; | ||||
| const AUDIO_ENABLED: bool = false; | ||||
|  | ||||
| fn main() -> Result<()> { | ||||
|     let app = App::new(crate_name!()) | ||||
| @@ -44,15 +41,6 @@ fn main() -> Result<()> { | ||||
|         ) | ||||
|         .get_matches(); | ||||
|  | ||||
|     // Set up subscriber | ||||
|     if std::env::var("RUST_LOG").is_err() { | ||||
|         std::env::set_var("RUST_LOG", "gb=info"); | ||||
|     } | ||||
|  | ||||
|     tracing_subscriber::fmt::fmt() | ||||
|         .with_env_filter(EnvFilter::from_default_env()) | ||||
|         .init(); | ||||
|  | ||||
|     let mut emu_build = | ||||
|         EmulatorBuilder::new().with_cart(m.value_of("rom").expect("ROM path provided"))?; | ||||
|  | ||||
| @@ -61,13 +49,8 @@ fn main() -> Result<()> { | ||||
|     } | ||||
|  | ||||
|     let mut emu = emu_build.finish(); | ||||
|  | ||||
|     // Load Save file if it exists | ||||
|     info!("Attempt to load .sav"); | ||||
|     emu.try_load_sav().expect("Load save if exists"); | ||||
|     let rom_title = emu.title(); | ||||
|  | ||||
|     info!("Initialize Gamepad"); | ||||
|     let mut gamepad = Gilrs::new().expect("Initialize Controller Support"); | ||||
|  | ||||
|     // Initialize GUI | ||||
| @@ -88,7 +71,8 @@ fn main() -> Result<()> { | ||||
|     let (_stream, stream_handle) = OutputStream::try_default().expect("Initialized Audio"); | ||||
|  | ||||
|     if AUDIO_ENABLED { | ||||
|         let (prod, cons) = gb::spsc_init(); | ||||
|         let spsc: AudioSPSC<f32> = Default::default(); | ||||
|         let (prod, cons) = spsc.init(); | ||||
|         let sink = { | ||||
|             let s = Sink::try_new(&stream_handle)?; | ||||
|             s.append(cons); | ||||
| @@ -98,7 +82,6 @@ fn main() -> Result<()> { | ||||
|  | ||||
|         emu.set_prod(prod); | ||||
|  | ||||
|         info!("Spawn Audio Thread"); | ||||
|         std::thread::spawn(move || { | ||||
|             sink.sleep_until_end(); | ||||
|         }); | ||||
| @@ -108,9 +91,11 @@ fn main() -> Result<()> { | ||||
|  | ||||
|     event_loop.run(move |event, _, control_flow| { | ||||
|         if let Event::RedrawRequested(_) = event { | ||||
|             if pixels.render().is_err() { | ||||
|                 emu.try_write_sav().expect("Write game save if need be"); | ||||
|  | ||||
|             if pixels | ||||
|                 .render() | ||||
|                 .map_err(|e| anyhow!("pixels.render() failed: {}", e)) | ||||
|                 .is_err() | ||||
|             { | ||||
|                 *control_flow = ControlFlow::Exit; | ||||
|                 return; | ||||
|             } | ||||
| @@ -118,8 +103,6 @@ fn main() -> Result<()> { | ||||
|  | ||||
|         if input.update(&event) { | ||||
|             if input.key_pressed(VirtualKeyCode::Escape) || input.quit() { | ||||
|                 emu.try_write_sav().expect("Write game save if need be"); | ||||
|  | ||||
|                 *control_flow = ControlFlow::Exit; | ||||
|                 return; | ||||
|             } | ||||
| @@ -130,8 +113,8 @@ fn main() -> Result<()> { | ||||
|  | ||||
|             cycle_count += gb::emu::run_frame(&mut emu, &mut gamepad, &input); | ||||
|  | ||||
|             if cycle_count >= CYCLES_IN_FRAME { | ||||
|                 cycle_count %= CYCLES_IN_FRAME; | ||||
|             if cycle_count >= gb::emu::CYCLES_IN_FRAME { | ||||
|                 cycle_count %= gb::emu::CYCLES_IN_FRAME; | ||||
|  | ||||
|                 let buf: &mut [u8; GB_WIDTH * GB_HEIGHT * 4] = pixels | ||||
|                     .get_frame() | ||||
|   | ||||
							
								
								
									
										286
									
								
								src/ppu.rs
									
									
									
									
									
								
							
							
						
						
									
										286
									
								
								src/ppu.rs
									
									
									
									
									
								
							| @@ -45,18 +45,18 @@ pub struct Ppu { | ||||
|     vram: Box<[u8; VRAM_SIZE]>, | ||||
|     pub(crate) oam: ObjectAttributeTable, | ||||
|     pub(crate) dma: DirectMemoryAccess, | ||||
|     scan_dot: Cycle, | ||||
|     scan_state: OamScanState, | ||||
|     fetch: PixelFetcher, | ||||
|     fifo: PixelFifo, | ||||
|     obj_buffer: ObjectBuffer, | ||||
|     frame_buf: Box<[u8; GB_WIDTH * GB_HEIGHT * 4]>, | ||||
|     win_stat: WindowStatus, | ||||
|     window_stat: WindowStatus, | ||||
|  | ||||
|     scanline_start: bool, | ||||
|     to_discard: u8, | ||||
|  | ||||
|     x_pos: u8, | ||||
|     dot: Cycle, | ||||
|     cycle: Cycle, | ||||
| } | ||||
|  | ||||
| impl BusIo for Ppu { | ||||
| @@ -71,7 +71,7 @@ impl BusIo for Ppu { | ||||
|  | ||||
| impl Ppu { | ||||
|     pub(crate) fn tick(&mut self) { | ||||
|         self.dot += 1; | ||||
|         self.cycle += 1; | ||||
|  | ||||
|         if !self.ctrl.lcd_enabled() { | ||||
|             return; | ||||
| @@ -79,15 +79,7 @@ impl Ppu { | ||||
|  | ||||
|         match self.stat.mode() { | ||||
|             PpuMode::OamScan => { | ||||
|                 // Cycles 1 -> 80 | ||||
|                 if self.dot >= 80 { | ||||
|                     self.x_pos = 0; | ||||
|                     self.scanline_start = true; | ||||
|                     self.fetch.back.scanline_first = true; | ||||
|                     self.to_discard = 0; | ||||
|                     self.fifo.back.clear(); | ||||
|                     self.fifo.obj.clear(); | ||||
|  | ||||
|                 if self.cycle >= 80 { | ||||
|                     self.stat.set_mode(PpuMode::Drawing); | ||||
|                 } | ||||
|  | ||||
| @@ -96,7 +88,7 @@ impl Ppu { | ||||
|             PpuMode::Drawing => { | ||||
|                 if self.ctrl.lcd_enabled() { | ||||
|                     // Only Draw when the LCD Is Enabled | ||||
|                     self.draw(); | ||||
|                     self.draw(self.cycle); | ||||
|                 } else { | ||||
|                     self.reset(); | ||||
|                 } | ||||
| @@ -112,22 +104,29 @@ impl Ppu { | ||||
|  | ||||
|                     // Increment Window line counter if scanline had any window pixels on it | ||||
|                     // only increment once per scanline though | ||||
|                     if self.win_stat.enabled { | ||||
|                         self.fetch.back.wl_count += 1; | ||||
|                     if self.window_stat.should_draw() { | ||||
|                         self.fetch.back.window_line.increment(); | ||||
|                     } | ||||
|  | ||||
|                     self.fetch.hblank_reset(); | ||||
|                     self.win_stat.enabled = false; | ||||
|                     self.x_pos = 0; | ||||
|                     self.scanline_start = true; | ||||
|                     self.to_discard = 0; | ||||
|  | ||||
|                     self.fetch.hblank_reset(); | ||||
|                     self.window_stat.hblank_reset(); | ||||
|                     self.obj_buffer.clear(); | ||||
|  | ||||
|                     self.fifo.back.clear(); | ||||
|                     self.fifo.obj.clear(); | ||||
|  | ||||
|                     self.stat.set_mode(PpuMode::HBlank); | ||||
|                 } | ||||
|             } | ||||
|             PpuMode::HBlank => { | ||||
|                 // This mode will always end at 456 cycles | ||||
|  | ||||
|                 if self.dot >= 456 { | ||||
|                     self.dot %= 456; | ||||
|                 if self.cycle >= 456 { | ||||
|                     self.cycle %= 456; | ||||
|                     self.pos.line_y += 1; | ||||
|  | ||||
|                     // Update LY==LYC bit | ||||
| @@ -144,10 +143,9 @@ impl Ppu { | ||||
|                         self.int.set_vblank(true); | ||||
|  | ||||
|                         // Reset Window Line Counter in Fetcher | ||||
|                         self.fetch.back.wl_count = 0; | ||||
|  | ||||
|                         self.fetch.vblank_reset(); | ||||
|                         // Reset WY=LY coincidence flag | ||||
|                         self.win_stat.coincidence = false; | ||||
|                         self.window_stat.vblank_reset(); | ||||
|  | ||||
|                         if self.stat.vblank_int() { | ||||
|                             // Enable Vblank LCDStat Interrupt | ||||
| @@ -161,7 +159,7 @@ impl Ppu { | ||||
|                             self.int.set_lcd_stat(true); | ||||
|                         } | ||||
|  | ||||
|                         self.scan_dot = Default::default(); | ||||
|                         self.scan_state.reset(); | ||||
|                         PpuMode::OamScan | ||||
|                     }; | ||||
|  | ||||
| @@ -169,8 +167,8 @@ impl Ppu { | ||||
|                 } | ||||
|             } | ||||
|             PpuMode::VBlank => { | ||||
|                 if self.dot >= 456 { | ||||
|                     self.dot %= 456; | ||||
|                 if self.cycle > 456 { | ||||
|                     self.cycle %= 456; | ||||
|                     self.pos.line_y += 1; | ||||
|  | ||||
|                     // Update LY==LYC bit | ||||
| @@ -190,7 +188,8 @@ impl Ppu { | ||||
|                             self.int.set_lcd_stat(true); | ||||
|                         } | ||||
|  | ||||
|                         self.scan_dot = Default::default(); | ||||
|                         self.scan_state.reset(); | ||||
|  | ||||
|                         self.stat.set_mode(PpuMode::OamScan); | ||||
|                     } | ||||
|                 } | ||||
| @@ -199,31 +198,37 @@ impl Ppu { | ||||
|     } | ||||
|  | ||||
|     fn scan_oam(&mut self) { | ||||
|         if self.scan_dot % 2 == 0 { | ||||
|             if self.dma.is_active() { | ||||
|                 return; | ||||
|         match self.scan_state.mode() { | ||||
|             OamScanMode::Scan if !self.dma.is_active() => { | ||||
|                 if !self.window_stat.coincidence() && self.scan_state.count() == 0 { | ||||
|                     // Determine whether we should draw the window next frame | ||||
|                     self.window_stat | ||||
|                         .set_coincidence(self.pos.line_y == self.pos.window_y); | ||||
|                 } | ||||
|  | ||||
|             if !self.win_stat.coincidence && self.scan_dot == 0 { | ||||
|                 self.win_stat.coincidence = self.pos.line_y == self.pos.window_y; | ||||
|             } | ||||
|                 let sprite_height = self.ctrl.obj_size().as_u8(); | ||||
|                 let index = self.scan_state.count(); | ||||
|  | ||||
|             let obj_height = self.ctrl.obj_size().size(); | ||||
|             let attr = self.oam.attribute(self.scan_dot as usize / 2); | ||||
|                 let attr = self.oam.attribute(index as usize); | ||||
|                 let line_y = self.pos.line_y + 16; | ||||
|  | ||||
|                 if attr.x > 0 | ||||
|                     && line_y >= attr.y | ||||
|                 && line_y < (attr.y + obj_height) | ||||
|                     && line_y < (attr.y + sprite_height) | ||||
|                     && !self.obj_buffer.is_full() | ||||
|                 { | ||||
|                     self.obj_buffer.add(attr); | ||||
|                 } | ||||
|  | ||||
|                 self.scan_state.increase(); | ||||
|             } | ||||
|         self.scan_dot += 1; | ||||
|             _ => {} | ||||
|         } | ||||
|  | ||||
|     fn draw(&mut self) { | ||||
|         self.scan_state.next(); | ||||
|     } | ||||
|  | ||||
|     fn draw(&mut self, _cycle: Cycle) { | ||||
|         use FetcherState::*; | ||||
|  | ||||
|         let mut iter = self.obj_buffer.iter_mut(); | ||||
| @@ -256,7 +261,7 @@ impl Ppu { | ||||
|                 TileLow => { | ||||
|                     let obj_size = self.ctrl.obj_size(); | ||||
|  | ||||
|                     let addr = PixelFetcher::get_obj_addr(attr, &self.pos, obj_size); | ||||
|                     let addr = PixelFetcher::get_obj_addr(&attr, &self.pos, obj_size); | ||||
|  | ||||
|                     let byte = self.read_byte(addr); | ||||
|                     self.fetch.obj.tile.with_low_byte(byte); | ||||
| @@ -267,7 +272,7 @@ impl Ppu { | ||||
|                 TileHigh => { | ||||
|                     let obj_size = self.ctrl.obj_size(); | ||||
|  | ||||
|                     let addr = PixelFetcher::get_obj_addr(attr, &self.pos, obj_size); | ||||
|                     let addr = PixelFetcher::get_obj_addr(&attr, &self.pos, obj_size); | ||||
|  | ||||
|                     let byte = self.read_byte(addr + 1); | ||||
|                     self.fetch.obj.tile.with_high_byte(byte); | ||||
| @@ -317,49 +322,60 @@ impl Ppu { | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         if self.ctrl.window_enabled() | ||||
|             && !self.window_stat.should_draw() | ||||
|             && self.window_stat.coincidence() | ||||
|             && self.x_pos as i16 >= self.pos.window_x as i16 - 7 | ||||
|         { | ||||
|             self.window_stat.set_should_draw(true); | ||||
|             self.fetch.back.reset(); | ||||
|             self.fetch.x_pos = 0; | ||||
|             self.fifo.back.clear(); | ||||
|         } | ||||
|  | ||||
|         if self.fetch.back.is_enabled() { | ||||
|             match self.fetch.back.state { | ||||
|                 SleepOne => self.fetch.back.next(TileNumber), | ||||
|                 TileNumber => { | ||||
|                     let x_pos = self.fetch.x_pos; | ||||
|  | ||||
|                     self.fetch.back.should_render_window(self.win_stat.enabled); | ||||
|                     self.fetch | ||||
|                         .back | ||||
|                         .should_render_window(self.window_stat.should_draw()); | ||||
|  | ||||
|                     let addr = self.fetch.bg_tile_num_addr(&self.ctrl, &self.pos, x_pos); | ||||
|  | ||||
|                     let id = self.read_byte(addr); | ||||
|                     self.fetch.back.tile.with_id(id); | ||||
|  | ||||
|                     self.fetch.back.next(TileLow); | ||||
|                     // Move on to the Next state in 2 T-cycles | ||||
|                     self.fetch.back.next(SleepOne); | ||||
|                 } | ||||
|                 SleepTwo => self.fetch.back.next(TileLow), | ||||
|                 SleepOne => self.fetch.back.next(TileLow), | ||||
|                 TileLow => { | ||||
|                     let addr = self.fetch.bg_byte_addr(&self.ctrl, &self.pos); | ||||
|  | ||||
|                     let low = self.read_byte(addr); | ||||
|                     self.fetch.back.tile.with_low_byte(low); | ||||
|  | ||||
|                     self.fetch.back.next(SleepThree); | ||||
|                     self.fetch.back.next(SleepTwo); | ||||
|                 } | ||||
|                 SleepThree => self.fetch.back.next(TileHigh), | ||||
|                 SleepTwo => self.fetch.back.next(TileHigh), | ||||
|                 TileHigh => { | ||||
|                     let addr = self.fetch.bg_byte_addr(&self.ctrl, &self.pos); | ||||
|  | ||||
|                     let high = self.read_byte(addr + 1); | ||||
|                     self.fetch.back.tile.with_high_byte(high); | ||||
|  | ||||
|                     if self.fetch.back.scanline_first { | ||||
|                         self.fetch.back.reset(); | ||||
|  | ||||
|                         self.fetch.back.scanline_first = false; | ||||
|                     } else { | ||||
|                         self.fetch.back.next(ToFifoOne); | ||||
|                     self.fetch.back.next(SleepThree); | ||||
|                 } | ||||
|                 SleepThree => self.fetch.back.next(ToFifoOne), | ||||
|                 ToFifoOne => { | ||||
|                     self.fetch.back.next(ToFifoTwo); | ||||
|                 } | ||||
|                 ToFifoOne | ToFifoTwo => { | ||||
|                 ToFifoTwo => { | ||||
|                     if let Ok(()) = self.fetch.send_to_fifo(&mut self.fifo) { | ||||
|                         self.fetch.x_pos += 1; | ||||
|                         self.fetch.back.next(SleepOne); | ||||
|                         self.fetch.back.next(TileNumber); | ||||
|                         self.fetch.back.tile = Default::default(); | ||||
|                     } | ||||
|                 } | ||||
| @@ -372,7 +388,7 @@ impl Ppu { | ||||
|                 self.scanline_start = false; | ||||
|             } | ||||
|  | ||||
|             if !self.win_stat.enabled && self.to_discard > 0 && !self.fifo.back.is_empty() { | ||||
|             if self.to_discard > 0 && !self.fifo.back.is_empty() { | ||||
|                 let _ = self.fifo.back.pop_front(); | ||||
|                 self.to_discard -= 1; | ||||
|  | ||||
| @@ -390,17 +406,6 @@ impl Ppu { | ||||
|  | ||||
|                 self.x_pos += 1; | ||||
|             } | ||||
|  | ||||
|             if self.ctrl.window_enabled() | ||||
|                 && !self.win_stat.enabled | ||||
|                 && self.win_stat.coincidence | ||||
|                 && self.x_pos as i16 >= self.pos.window_x as i16 - 7 | ||||
|             { | ||||
|                 self.win_stat.enabled = true; | ||||
|                 self.fetch.back.reset(); | ||||
|                 self.fetch.x_pos = 0; | ||||
|                 self.fifo.back.clear(); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -408,11 +413,11 @@ impl Ppu { | ||||
|         self.pos.line_y = 0; | ||||
|         self.stat.set_mode(PpuMode::HBlank); | ||||
|  | ||||
|         // TODO: Is this an unnecessary performance hit? | ||||
|         let mut blank = WHITE.repeat(self.frame_buf.len() / 4); | ||||
|         self.frame_buf.swap_with_slice(&mut blank); | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     pub(crate) fn frame_buf(&self) -> &[u8; GB_HEIGHT * GB_WIDTH * 4] { | ||||
|         &self.frame_buf | ||||
|     } | ||||
| @@ -465,7 +470,7 @@ impl Default for Ppu { | ||||
|     fn default() -> Self { | ||||
|         Self { | ||||
|             vram: Box::new([0u8; VRAM_SIZE]), | ||||
|             dot: Default::default(), | ||||
|             cycle: Default::default(), | ||||
|             frame_buf: Box::new([0; GB_WIDTH * GB_HEIGHT * 4]), | ||||
|             int: Default::default(), | ||||
|             ctrl: Default::default(), | ||||
| @@ -473,11 +478,11 @@ impl Default for Ppu { | ||||
|             pos: Default::default(), | ||||
|             stat: Default::default(), | ||||
|             oam: Default::default(), | ||||
|             scan_dot: Default::default(), | ||||
|             scan_state: Default::default(), | ||||
|             fetch: Default::default(), | ||||
|             fifo: Default::default(), | ||||
|             obj_buffer: Default::default(), | ||||
|             win_stat: Default::default(), | ||||
|             window_stat: Default::default(), | ||||
|             dma: Default::default(), | ||||
|             x_pos: 0, | ||||
|             scanline_start: true, | ||||
| @@ -624,7 +629,6 @@ impl ObjectBuffer { | ||||
|         self.len += 1; | ||||
|     } | ||||
|  | ||||
|     #[inline] | ||||
|     fn iter_mut(&mut self) -> std::slice::IterMut<'_, Option<ObjectAttribute>> { | ||||
|         self.inner.iter_mut() | ||||
|     } | ||||
| @@ -653,6 +657,10 @@ impl PixelFetcher { | ||||
|         self.x_pos = 0; | ||||
|     } | ||||
|  | ||||
|     fn vblank_reset(&mut self) { | ||||
|         self.back.vblank_reset(); | ||||
|     } | ||||
|  | ||||
|     fn bg_tile_num_addr(&self, control: &LCDControl, pos: &ScreenPosition, x_pos: u8) -> u16 { | ||||
|         let line_y = pos.line_y; | ||||
|         let scroll_y = pos.scroll_y; | ||||
| @@ -672,7 +680,7 @@ impl PixelFetcher { | ||||
|  | ||||
|         let scx_offset = if is_window { 0 } else { scroll_x / 8 }; | ||||
|         let y_offset = if is_window { | ||||
|             self.back.wl_count as u16 / 8 | ||||
|             self.back.window_line.count() as u16 / 8 | ||||
|         } else { | ||||
|             ((line_y as u16 + scroll_y as u16) & 0xFF) / 8 | ||||
|         }; | ||||
| @@ -691,12 +699,12 @@ impl PixelFetcher { | ||||
|         let id = self.back.tile.id.expect("Tile Number is present"); | ||||
|  | ||||
|         let tile_data_addr = match control.tile_data_addr() { | ||||
|             TileDataAddress::X8800 => 0x9000u16.wrapping_add((id as i8 as i16 * 16) as u16), | ||||
|             TileDataAddress::X8800 => 0x9000u16.wrapping_add(((id as i8) as i16 * 16) as u16), | ||||
|             TileDataAddress::X8000 => 0x8000 + (id as u16 * 16), | ||||
|         }; | ||||
|  | ||||
|         let offset = if is_window { | ||||
|             self.back.wl_count as u16 % 8 | ||||
|             self.back.window_line.count() as u16 % 8 | ||||
|         } else { | ||||
|             (line_y as u16 + scroll_y as u16) % 8 | ||||
|         }; | ||||
| @@ -758,10 +766,9 @@ trait Fetcher { | ||||
| struct BackgroundFetcher { | ||||
|     state: FetcherState, | ||||
|     tile: TileBuilder, | ||||
|     wl_count: u8, | ||||
|     window_line: WindowLineCounter, | ||||
|     is_window_tile: bool, | ||||
|     enabled: bool, | ||||
|     scanline_first: bool, | ||||
| } | ||||
|  | ||||
| impl BackgroundFetcher { | ||||
| @@ -784,6 +791,10 @@ impl BackgroundFetcher { | ||||
|     fn is_enabled(&self) -> bool { | ||||
|         self.enabled | ||||
|     } | ||||
|  | ||||
|     fn vblank_reset(&mut self) { | ||||
|         self.window_line.vblank_reset(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Fetcher for BackgroundFetcher { | ||||
| @@ -792,7 +803,7 @@ impl Fetcher for BackgroundFetcher { | ||||
|     } | ||||
|  | ||||
|     fn reset(&mut self) { | ||||
|         self.state = FetcherState::SleepOne; | ||||
|         self.state = Default::default(); | ||||
|         self.tile = Default::default(); | ||||
|     } | ||||
|  | ||||
| @@ -808,43 +819,53 @@ impl Fetcher for BackgroundFetcher { | ||||
| impl Default for BackgroundFetcher { | ||||
|     fn default() -> Self { | ||||
|         Self { | ||||
|             state: FetcherState::SleepOne, | ||||
|             state: Default::default(), | ||||
|             tile: Default::default(), | ||||
|             is_window_tile: Default::default(), | ||||
|             wl_count: Default::default(), | ||||
|             window_line: Default::default(), | ||||
|             enabled: true, | ||||
|             scanline_first: true, | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug)] | ||||
| #[derive(Debug, Default)] | ||||
| struct ObjectFetcher { | ||||
|     state: FetcherState, | ||||
|     tile: TileBuilder, | ||||
| } | ||||
|  | ||||
| impl Default for ObjectFetcher { | ||||
|     fn default() -> Self { | ||||
|         Self { | ||||
|             state: FetcherState::TileNumber, | ||||
|             tile: Default::default(), | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| impl Fetcher for ObjectFetcher { | ||||
|     fn next(&mut self, state: FetcherState) { | ||||
|         self.state = state | ||||
|     } | ||||
|  | ||||
|     fn reset(&mut self) { | ||||
|         self.state = FetcherState::TileNumber; | ||||
|         self.state = Default::default(); | ||||
|         self.tile = Default::default(); | ||||
|     } | ||||
|  | ||||
|     fn hblank_reset(&mut self) { | ||||
|         self.reset() | ||||
|         self.state = Default::default(); | ||||
|         self.tile = Default::default(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug, Default)] | ||||
| struct WindowLineCounter { | ||||
|     count: u8, | ||||
| } | ||||
|  | ||||
| impl WindowLineCounter { | ||||
|     fn increment(&mut self) { | ||||
|         self.count += 1; | ||||
|     } | ||||
|  | ||||
|     fn vblank_reset(&mut self) { | ||||
|         self.count = 0; | ||||
|     } | ||||
|  | ||||
|     fn count(&self) -> u8 { | ||||
|         self.count | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -860,6 +881,12 @@ enum FetcherState { | ||||
|     ToFifoTwo, | ||||
| } | ||||
|  | ||||
| impl Default for FetcherState { | ||||
|     fn default() -> Self { | ||||
|         Self::TileNumber | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug, Default)] | ||||
| struct BgPixelProperty { | ||||
|     shade_id: u8, | ||||
| @@ -930,11 +957,84 @@ impl TileBuilder { | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug, Default)] | ||||
| struct OamScanState { | ||||
|     count: u8, | ||||
|     mode: OamScanMode, | ||||
| } | ||||
|  | ||||
| impl OamScanState { | ||||
|     fn increase(&mut self) { | ||||
|         self.count += 1; | ||||
|         self.count %= 40; | ||||
|     } | ||||
|  | ||||
|     fn reset(&mut self) { | ||||
|         self.count = Default::default(); | ||||
|         self.mode = Default::default(); | ||||
|     } | ||||
|  | ||||
|     fn count(&self) -> u8 { | ||||
|         self.count | ||||
|     } | ||||
|  | ||||
|     fn mode(&self) -> &OamScanMode { | ||||
|         &self.mode | ||||
|     } | ||||
|  | ||||
|     fn next(&mut self) { | ||||
|         use OamScanMode::*; | ||||
|  | ||||
|         self.mode = match self.mode { | ||||
|             Scan => Sleep, | ||||
|             Sleep => Scan, | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug, Clone, Copy, PartialEq)] | ||||
| enum OamScanMode { | ||||
|     Scan, | ||||
|     Sleep, | ||||
| } | ||||
|  | ||||
| impl Default for OamScanMode { | ||||
|     fn default() -> Self { | ||||
|         Self::Scan | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Debug, Default)] | ||||
| struct WindowStatus { | ||||
|     /// This will be true if WY == LY at any point in the frame thus far | ||||
|     coincidence: bool, | ||||
|     /// This will be true if the conditions which tell the PPU to start | ||||
|     /// drawing from the window tile map is true | ||||
|     enabled: bool, | ||||
|     should_draw: bool, | ||||
| } | ||||
|  | ||||
| impl WindowStatus { | ||||
|     fn should_draw(&self) -> bool { | ||||
|         self.should_draw | ||||
|     } | ||||
|  | ||||
|     fn coincidence(&self) -> bool { | ||||
|         self.coincidence | ||||
|     } | ||||
|  | ||||
|     fn set_should_draw(&mut self, value: bool) { | ||||
|         self.should_draw = value; | ||||
|     } | ||||
|  | ||||
|     fn set_coincidence(&mut self, value: bool) { | ||||
|         self.coincidence = value; | ||||
|     } | ||||
|  | ||||
|     fn hblank_reset(&mut self) { | ||||
|         self.should_draw = false; | ||||
|     } | ||||
|  | ||||
|     fn vblank_reset(&mut self) { | ||||
|         self.coincidence = false; | ||||
|     } | ||||
| } | ||||
|   | ||||
| @@ -183,7 +183,7 @@ pub enum ObjectSize { | ||||
| } | ||||
|  | ||||
| impl ObjectSize { | ||||
|     pub(crate) fn size(&self) -> u8 { | ||||
|     pub(crate) fn as_u8(&self) -> u8 { | ||||
|         use ObjectSize::*; | ||||
|  | ||||
|         match self { | ||||
|   | ||||
							
								
								
									
										31
									
								
								src/timer.rs
									
									
									
									
									
								
							
							
						
						
									
										31
									
								
								src/timer.rs
									
									
									
									
									
								
							| @@ -21,6 +21,17 @@ impl Timer { | ||||
|         use State::*; | ||||
|         use TimerSpeed::*; | ||||
|  | ||||
|         match self.state { | ||||
|             TIMAOverflow(_) | AbortedTIMAOverflow(_) => self.next(), | ||||
|             LoadTMA => { | ||||
|                 self.counter = self.modulo; | ||||
|                 self.interrupt = true; | ||||
|  | ||||
|                 self.next(); | ||||
|             } | ||||
|             Normal => {} | ||||
|         } | ||||
|  | ||||
|         self.divider = self.divider.wrapping_add(1); | ||||
|  | ||||
|         // Get Bit Position | ||||
| @@ -42,17 +53,6 @@ impl Timer { | ||||
|         } | ||||
|  | ||||
|         self.and_result = Some(new_result); | ||||
|  | ||||
|         match self.state { | ||||
|             TIMAOverflow(_) | AbortedTIMAOverflow(_) => self.next(), | ||||
|             LoadTIMA => { | ||||
|                 self.counter = self.modulo; | ||||
|                 self.interrupt = true; | ||||
|  | ||||
|                 self.next(); | ||||
|             } | ||||
|             Normal => {} | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     /// 0xFF05 | TIMA - Timer Counter | ||||
| @@ -70,7 +70,7 @@ impl Timer { | ||||
|                 self.counter = byte; | ||||
|                 self.state = AbortedTIMAOverflow(step); | ||||
|             } | ||||
|             LoadTIMA => { /* Ignored */ } | ||||
|             LoadTMA => {} | ||||
|         } | ||||
|     } | ||||
|  | ||||
| @@ -95,8 +95,9 @@ impl Timer { | ||||
|         use State::*; | ||||
|  | ||||
|         self.state = match self.state { | ||||
|             Normal | LoadTIMA | AbortedTIMAOverflow(3) => Normal, | ||||
|             TIMAOverflow(3) => LoadTIMA, | ||||
|             Normal | LoadTMA => Normal, | ||||
|             AbortedTIMAOverflow(4) => Normal, | ||||
|             TIMAOverflow(4) => LoadTMA, | ||||
|             AbortedTIMAOverflow(step) => AbortedTIMAOverflow(step + 1), | ||||
|             TIMAOverflow(step) => TIMAOverflow(step + 1), | ||||
|         } | ||||
| @@ -180,5 +181,5 @@ enum State { | ||||
|     TIMAOverflow(u8), | ||||
|     AbortedTIMAOverflow(u8), | ||||
|     Normal, | ||||
|     LoadTIMA, | ||||
|     LoadTMA, | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user