| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-29 19:00:09 -04:00
										 |  |  | #include <cstring>
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							|  |  |  | #include "core/crypto/ctr_encryption_layer.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | namespace Core::Crypto { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | CTREncryptionLayer::CTREncryptionLayer(FileSys::VirtualFile base_, Key128 key_, size_t base_offset) | 
					
						
							|  |  |  |     : EncryptionLayer(std::move(base_)), base_offset(base_offset), cipher(key_, Mode::CTR), | 
					
						
							|  |  |  |       iv(16, 0) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | size_t CTREncryptionLayer::Read(u8* data, size_t length, size_t offset) const { | 
					
						
							|  |  |  |     if (length == 0) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const auto sector_offset = offset & 0xF; | 
					
						
							|  |  |  |     if (sector_offset == 0) { | 
					
						
							|  |  |  |         UpdateIV(base_offset + offset); | 
					
						
							|  |  |  |         std::vector<u8> raw = base->ReadBytes(length, offset); | 
					
						
							|  |  |  |         if (raw.size() != length) | 
					
						
							|  |  |  |             return Read(data, raw.size(), offset); | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |         cipher.Transcode(raw.data(), length, data, Op::Decrypt); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |         return length; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // offset does not fall on block boundary (0x10)
 | 
					
						
							|  |  |  |     std::vector<u8> block = base->ReadBytes(0x10, offset - sector_offset); | 
					
						
							|  |  |  |     UpdateIV(base_offset + offset - sector_offset); | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  |     cipher.Transcode(block.data(), block.size(), block.data(), Op::Decrypt); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     size_t read = 0x10 - sector_offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (length + sector_offset < 0x10) { | 
					
						
							| 
									
										
										
										
											2018-07-29 19:00:09 -04:00
										 |  |  |         std::memcpy(data, block.data() + sector_offset, std::min<u64>(length, read)); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |         return read; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-29 19:00:09 -04:00
										 |  |  |     std::memcpy(data, block.data() + sector_offset, read); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  |     return read + Read(data + read, length - read, offset + read); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | void CTREncryptionLayer::SetIV(const std::vector<u8>& iv_) { | 
					
						
							|  |  |  |     const auto length = std::min(iv_.size(), iv.size()); | 
					
						
							|  |  |  |     iv.assign(iv_.cbegin(), iv_.cbegin() + length); | 
					
						
							| 
									
										
										
										
											2018-07-27 23:55:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CTREncryptionLayer::UpdateIV(size_t offset) const { | 
					
						
							|  |  |  |     offset >>= 4; | 
					
						
							|  |  |  |     for (size_t i = 0; i < 8; ++i) { | 
					
						
							|  |  |  |         iv[16 - i - 1] = offset & 0xFF; | 
					
						
							|  |  |  |         offset >>= 8; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     cipher.SetIV(iv); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-07-28 21:39:42 -04:00
										 |  |  | } // namespace Core::Crypto
 |