forked from eden-emu/eden
		
	 99ceb03a1c
			
		
	
	
		99ceb03a1c
		
	
	
	
	
		
			
			This formats all copyright comments according to SPDX formatting guidelines. Additionally, this resolves the remaining GPLv2 only licensed files by relicensing them to GPLv2.0-or-later.
		
			
				
	
	
		
			52 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			52 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <cstring>
 | |
| #include "core/crypto/ctr_encryption_layer.h"
 | |
| 
 | |
| namespace Core::Crypto {
 | |
| 
 | |
| CTREncryptionLayer::CTREncryptionLayer(FileSys::VirtualFile base_, Key128 key_,
 | |
|                                        std::size_t base_offset_)
 | |
|     : EncryptionLayer(std::move(base_)), base_offset(base_offset_), cipher(key_, Mode::CTR) {}
 | |
| 
 | |
| std::size_t CTREncryptionLayer::Read(u8* data, std::size_t length, std::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);
 | |
|         cipher.Transcode(raw.data(), raw.size(), data, Op::Decrypt);
 | |
|         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);
 | |
|     cipher.Transcode(block.data(), block.size(), block.data(), Op::Decrypt);
 | |
|     std::size_t read = 0x10 - sector_offset;
 | |
| 
 | |
|     if (length + sector_offset < 0x10) {
 | |
|         std::memcpy(data, block.data() + sector_offset, std::min<u64>(length, read));
 | |
|         return std::min<u64>(length, read);
 | |
|     }
 | |
|     std::memcpy(data, block.data() + sector_offset, read);
 | |
|     return read + Read(data + read, length - read, offset + read);
 | |
| }
 | |
| 
 | |
| void CTREncryptionLayer::SetIV(const IVData& iv_) {
 | |
|     iv = iv_;
 | |
| }
 | |
| 
 | |
| void CTREncryptionLayer::UpdateIV(std::size_t offset) const {
 | |
|     offset >>= 4;
 | |
|     for (std::size_t i = 0; i < 8; ++i) {
 | |
|         iv[16 - i - 1] = offset & 0xFF;
 | |
|         offset >>= 8;
 | |
|     }
 | |
|     cipher.SetIV(iv);
 | |
| }
 | |
| } // namespace Core::Crypto
 |