VeraCrypt
aboutsummaryrefslogtreecommitdiff
path: root/doc/html/flattr-badge-large.png
blob: cbf1bbba08ccb96ae44e304cb220a503b56ca314 (plain)
ofshex dumpascii
0000 89 50 4e 47 0d 0a 1a 0a 00 00 00 0d 49 48 44 52 00 00 00 5d 00 00 00 14 08 06 00 00 00 ef 26 31 .PNG........IHDR...]..........&1
0020 67 00 00 00 19 74 45 58 74 53 6f 66 74 77 61 72 65 00 41 64 6f 62 65 20 49 6d 61 67 65 52 65 61 g....tEXtSoftware.Adobe.ImageRea
0040 64 79 71 c9 65 3c 00 00 03 23 69 54 58 74 58 4d 4c 3a 63 6f 6d 2e 61 64 6f 62 65 2e 78 6d 70 00 dyq.e<...#iTXtXML:com.adobe.xmp.
0060 00 00 00 00 3c 3f 78 70 61 63 6b 65 74 20 62 65 67 69 6e 3d 22 ef bb bf 22 20 69 64 3d 22 57 35 ....<?xpacket.begin="...".id="W5
0080 4d 30 4d 70 43 65 68 69 48 7a 72 65 53 7a 4e 54 63 7a 6b 63 39 64 22 3f 3e 20 3c 78 3a 78 6d 70 M0MpCehiHzreSzNTczkc9d"?>.<x:xmp
00a0 6d 65 74 61 20 78 6d 6c 6e 73 3a 78 3d 22 61 64 6f 62 65 3a 6e 73 3a 6d 65 74 61 2f 22 20 78 3a meta.xmlns:x="adobe:ns:meta/".x:
00c0 78 6d 70 74 6b 3d 22 41 64 6f 62 65 20 58 4d 50 20 43 6f 72 65 20 35 2e 35 2d 63 30 31 34 20 37 xmptk="Adobe.XMP.Core.5.5-c014.7
00e0 39 2e 31 35 31 34 38 31 2c 20 32 30 31 33 2f 30 33 2f 31 33 2d 31 32 3a 30 39 3a 31 35 20 20 20 9.151481,.2013/03/13-12:09:15...
0100 20 20 20 20 20 22 3e 20 3c 72 64 66 3a 52 44 46 20 78 6d 6c 6e 73 3a 72 64 66 3d 22 68 74 74 70 .....">.<rdf:RDF.xmlns:rdf="http
0120 3a 2f 2f 77 77 77 2e 77 33 2e 6f 72 67 2f 31 39 39 39 2f 30 32 2f 32 32 2d 72 64 66 2d 73 79 6e ://www.w3.org/1999/02/22-rdf-syn
0140 74 61 78 2d 6e 73 23 22 3e 20 3c 72 64 66 3a 44 65 73 63 72 69 70 74 69 6f 6e 20 72 64 66 3a 61 tax-ns#">.<rdf:Description.rdf:a
0160 62 6f 75 74 3d 22 22 20 78 6d 6c 6e 73 3a 78 6d 70 3d 22 68 74 74 70 3a 2f 2f 6e 73 2e 61 64 6f bout="".xmlns:xmp="http://ns.ado
0180 62 65 2e 63 6f 6d 2f 78 61 70 2f 31 2e 30 2f 22 20 78 6d 6c 6e 73 3a 78 6d 70 4d 4d 3d 22 68 74 be.com/xap/1.0/".xmlns:xmpMM="ht
01a0 74 70 3a 2f 2f 6e 73 2e 61 64 6f 62 65 2e 63 6f 6d 2f 78 61 70 2f 31 2e 30 2f 6d 6d 2f 22 20 78 tp://ns.adobe.com/xap/1.0/mm/".x
01c0 6d 6c 6e 73 3a 73 74 52 65 66 3d 22 68 74 74 70 3a 2f 2f 6e 73 2e 61 64 6f 62 65 2e 63 6f 6d 2f mlns:stRef="http://ns.adobe.com/
01e0 78 61 70 2f 31 2e 30 2f 73 54 79 70 65 2f 52 65 73 6f 75 72 63 65 52 65 66 23 22 20 78 6d 70 3a xap/1.0/sType/ResourceRef#".xmp:
0200 43 72 65 61 74 6f 72 54 6f 6f 6c 3d 22 41 64 6f 62 65 20 50 68 6f 74 6f 73 68 6f 70 20 43 43 20 CreatorTool="Adobe.Photoshop.CC.
0220 28 4d 61 63 69 6e 74 6f 73 68 29 22 20 78 6d 70 4d 4d 3a 49 6e 73 74 61 6e 63 65 49 44 3d 22 78 (Macintosh)".xmpMM:InstanceID="x
0240 6d 70 2e 69 69 64 3a 30 42 35 42 41 33 38 37 42 31 46 46 31 31 45 35 42 46 31 38 44 35 30 32 38 mp.iid:0B5BA387B1FF11E5BF18D5028
0260 33 31 33 31 31 36 43 22 20 78 6d 70 4d 4d 3a 44 6f 63 75 6d 65 6e 74 49 44 3d 22 78 6d 70 2e 64 313116C".xmpMM:DocumentID="xmp.d
0280 69 64 3a 30 42 35 42 41 33 38 38 42 31 46 46 31 31 45 35 42 46 31 38 44 35 30 32 38 33 31 33 31 id:0B5BA388B1FF11E5BF18D50283131
02a0 31 36 43 22 3e 20 3c 78 6d 70 4d 4d 3a 44 65 72 69 76 65 64 46 72 6f 6d 20 73 74 52 65 66 3a 69 16C">.<xmpMM:DerivedFrom.stRef:i
02c0 6e 73 74 61 6e 63 65 49 44 3d 22 78 6d 70 2e 69 69 64 3a 30 42 35 42 41 33 38 35 42 31 46 46 31 nstanceID="xmp.iid:0B5BA385B1FF1
02e0 31 45 35 42 46 31 38 44 35 30 32 38 33 31 33 31 31 36 43 22 20 73 74 52 65 66 3a 64 6f 63 75 6d 1E5BF18D5028313116C".stRef:docum
0300 65 6e 74 49 44 3d 22 78 6d 70 2e 64 69 64 3a 30 42 35 42 41 33 38 36 42 31 46 46 31 31 45 35 42 entID="xmp.did:0B5BA386B1FF11E5B
0320 46 31 38 44 35 30 32 38 33 31 33 31 31 36 43 22 2f 3e 20 3c 2f 72 64 66 3a 44 65 73 63 72 69 70 F18D5028313116C"/>.</rdf:Descrip
0340 74 69 6f 6e 3e 20 3c 2f 72 64 66 3a 52 44 46 3e 20 3c 2f 78 3a 78 6d 70 6d 65 74 61 3e 20 3c 3f tion>.</rdf:RDF>.</x:xmpmeta>.<?
0360 78 70 61 63 6b 65 74 20 65 6e 64 3d 22 72 22 3f 3e b8 b0 f5 d2 00 00 05 31 49 44 41 54 78 da ec xpacket.end="r"?>.......1IDATx..
0380 58 7b 4c 5b 55 18 ff 4e 7b 5b 0b ac c5 15 85 ad 73 9a ec c1 23 86 f4 0f 60 53 41 c7 96 f0 07 32 X{L[U..N{[......s...#...`SA....2
03a0 34 21 31 c6 34 21 be 22 f8 07 e2 58 44 25 31 20 ea 62 50 91 08 ea 40 8d 91 a4 6b b2 20 7f ac 6e 4!1.4!."...XD%1..bP...@...k....n
03c0 2e 21 4a 97 80 06 99 6e ac 26 50 82 ba 0d c6 20 b6 44 ca 53 da 7b 8f e7 9c f6 96 d6 7b bb 95 f5 .!J....n.&P......D.S.{......{...
03e0 12 1f e9 97 9c de 73 be ef 3b 8f fe be c7 f9 ee 45 af 9c 39 a4 45 58 d5 84 91 f0 24 00 6c 87 04 ......s..;......E..9.EX....$.l..
0400 6d 0a 21 50 cd 20 bf e6 a4 f6 72 f6 5b 1c 60 f4 46 4a 8a be fe c8 03 9f 43 92 c6 40 84 28 a4 88 m.!P......r.[.`.FJ......C..@.(..
0420 31 a6 bf 20 00 cf c6 6a a4 89 71 0b cc e6 7e e4 ac 4e a0 1d 24 9e f7 6f 5b f0 fe f1 c2 ca 8e 09 1......j..q...~..N..$..o[.......
0440 23 87 05 e1 e9 da fb 3f 81 64 4d aa d4 3a 08 31 1b a9 41 75 cb 9b 4d 2d ba 12 88 07 49 ad d5 80 #......?.dM..:.1..Au..M-....I...
0460 9f 83 47 39 5e c0 77 a4 68 b6 46 55 1c 19 19 01 bb dd 0e a3 a3 a3 b0 ba ba 2a 91 6b b5 5a 68 68 ..G9^.w.h.FU.............*.k.Zhh
0480 68 80 dc dc dc 88 60 42 48 8c 96 04 d8 22 f9 b1 8f e4 0c 95 9e c3 7e cc 40 92 28 f8 fd d0 de de h.....`BH...."........~.@.(.....
04a0 0e 03 03 03 51 17 51 a9 54 50 57 57 c7 00 f7 7a bd 60 30 18 a4 d1 22 a0 4d fd 23 cf e4 be 07 d7 ....Q.Q.TPWW...z.`0...".M.#.....
04c0 48 34 9d fd ad eb 3f 01 3c c5 9b c3 51 5c 51 04 5c ad 56 43 45 45 05 14 17 17 43 7a 7a 3a 74 75 H4....?.<...Q\Q.\.VCEE....Czz:tu
04e0 75 41 5f 5f 1f 03 bc be be 1e 0a 0a 0a c0 ed 76 43 53 53 13 74 74 74 48 37 89 d3 d5 df 7e e8 9c uA__...........vCSS.tttH7....~..
0500 2c df 3a da 08 ce df 1d b0 e7 f6 3c 7a 85 c4 bc 8f 25 e7 75 d8 a1 cf 84 96 1f 9e 88 e0 d7 ef b3 ,.:........<z....%.u............
0520 c1 b5 85 71 b6 ee a6 82 4e 8e c9 09 02 0e a5 82 f0 94 22 02 de dc dc 0c d9 d9 d9 8c df d3 d3 23 ...q....N........."............#
0540 01 9c 1d 98 f4 e7 e7 e7 e5 2f 10 05 0e ea 59 9d 86 a1 eb f6 08 9e 7b e5 7a 68 6d bc 81 7d 74 9c ........./....Y.......{.zhm..}t.
0560 1e 8c 3a 93 44 9f f2 3c 61 6b 6e 16 51 bc 09 e8 52 01 cd e1 94 a8 87 8b 80 53 5a 5e 5e 66 86 a8 ..:.D..<akn.Q...R........SZ^^f..
0580 ad ad 0d 01 4e 29 1a e0 2c 05 09 f1 1f 74 8e 80 71 ee ca 09 e9 da 32 fb ec dd 9a 07 a5 bb aa 88 ....N)..,....t..q.....2.........
05a0 37 67 91 79 01 63 39 ae 06 e6 be bc df 06 c6 24 13 eb bf 43 22 e8 eb 5f 3b e1 12 89 16 ca 17 e7 7g.y.c9........$...C".._;.......
05c0 52 fe a7 23 75 70 e0 6e 0b 1b 0f 4d db 61 bf a9 9c e9 8a eb c4 07 3a 39 ab 5c 58 d2 4b 93 12 4d R..#up.n...M.a........:9.\X.K..M
05e0 29 e1 54 59 59 09 56 ab 15 0a 0b 0b 63 df 84 ac 1f 4f 13 7d 39 16 b9 69 4b 26 3c 6b 6e 65 9e ff ).TYY.V.....c....O.}9..iK&<kne..
0600 a5 eb 5d 16 21 d4 00 fb b6 97 33 79 3f 01 cd 43 0c 41 e9 cc 2f 9d 30 e9 75 c1 d2 9a 97 f5 59 44 ..].!.....3y?..C.A../.0.u.....YD
0620 11 19 ed bb 99 4e 60 6d 9a 8a fa af 5a 99 6e bc ff 85 36 8a 37 f1 74 a9 2b 8a 55 4a 46 46 86 44 .....N`m....Z.n...6.7.t.+.UJFF.D
0640 a6 d1 68 36 6c d9 78 89 86 7e f1 4e 4b 04 ef db 2b d6 88 3c 49 f7 d1 a9 f5 30 3e 77 1e 4e 4f 74 ..h6l.x..~.NK...+..<I....0>w.NOt
0660 c2 d4 82 0b 2e cc f4 c3 9b 07 4e 43 ee 9d c5 f0 dd d4 29 d6 68 3f 8d 78 fb 37 97 d7 e7 d3 fe c3 ..........NC......).h?.x.7......
0680 bb ab 18 e8 22 5f b4 67 b7 b3 31 64 28 65 d2 8b 40 2e 52 3e b0 01 92 29 32 10 42 31 5c 0c 38 ea ...."_.g..1d(e..@.R>...)2.B1\.8.
06a0 3b 98 52 25 23 05 a9 6c 4f 55 04 2f 1c 34 71 1f 97 e7 3c 6b 77 19 b2 48 6a c8 97 c8 6f 34 8e c6 ;.R%#..lOU./.4q...<kw..Hj...o4..
06c0 77 2f 4f 2b 9a d3 29 de e4 e5 08 c7 04 2e 25 9f cf 27 f1 f4 d9 d9 d9 c0 05 a5 d3 49 de 4a 19 f4 w/O+..).......%..'.........I.J..
06e0 0a a0 4e bd b7 f3 c7 23 32 26 0d 1b 93 7d a8 71 aa f2 de 67 4f 90 91 df 68 bc 51 bd 5b 06 9d e0 ..N....#2&...}.q...gO...h.Q.[...
0700 ad a2 b7 69 2c 34 38 38 08 16 8b 05 ba bb bb 23 f8 0e 87 83 3d 73 72 72 fe f1 1a b8 6c 6f 35 03 ...i,488.......#....=srr....lo5.
0720 fc d8 c0 e3 70 b4 ef 41 d6 fe 6d 14 ac 5e 84 a0 57 46 fa 4e 6a ea fa 67 81 e1 e1 61 68 6b 6b 63 ....p..A..m..^..WF.Nj..g...ahkkc
0740 f9 28 39 39 39 c4 1f 1b 1b 83 de de 5e d6 2f 2f 2f 97 4f 2f 4a d4 60 f8 26 eb 04 e5 cb 6b 0b 6c .(999.......^.///.O/J.`.&....k.l
0760 78 f0 1e 0b 4c 91 8b 2f 33 2d 5f 32 7f 25 a8 53 b6 bb 1a 2e 92 9c 4f f5 18 df b7 00 3b 49 c5 73 x...L../3-_2.%.S......O.....;I.s
0780 9f e9 11 18 27 29 2a 18 a8 ca 9c 5f 92 d3 65 c2 c7 64 32 41 63 63 63 08 f0 96 96 16 a6 5c 52 52 ....')*...._..e..d2Accc......\RR
07a0 c2 ca c8 99 99 19 e6 e1 14 70 9e e7 a1 a8 a8 08 cc 66 b3 6c ae c7 0a 85 e7 cd d6 a1 f2 af 5c 1f .........p.......f.l..........\.
07c0 43 92 46 0f e6 6d 07 59 fb 7e f2 14 78 82 39 59 9c 6f 27 3a b4 6c 3c b4 cb c2 70 9d 9c 1f 63 fc C.F..m.Y.~..x.9Y.o':.l<...p...c.
07e0 93 3f b7 c0 e1 ac e7 e1 b1 7b 5f 82 2f 2e bc a6 d8 cb 9d dc 39 d1 53 c7 f3 f1 67 cf 0d 91 bc be .?.......{_./.......9.S...g.....
0800 5e f5 8a af f4 e1 80 47 23 0a 78 4d 4d 0d 70 1c 27 71 bf 0f 2f 55 c1 c4 d4 68 e2 a3 4b 18 cd 4f ^......G#.xMM.p.'q../U...h..K..O
0820 ae 02 07 3c 8e 00 9c 12 05 dc e9 74 42 6b 6b ab 2c e0 f4 d2 a4 39 9c a6 94 bf 7b b8 d2 9f 01 fe ...<.......tBkk.,....9....{.....
0840 7f df 78 59 4e 47 73 8b 6b 1e e3 16 6d 5a 84 8c 7e c4 b2 d9 6c 71 ac 8e c2 8b 98 04 05 2b 17 2c ..xYNGs.k...mZ..~...lq.......+.,
0860 a8 97 54 88 57 db 5e 3c 51 0a 04 f8 40 59 28 fc 49 bc 53 50 00 2d 1c aa 7b 13 0d c0 bf 86 61 69 ..T.W.^<Q...@Y(.I.SP.-..{.....ai
0880 ce 07 ba 35 e3 4f e8 d5 63 35 e9 b3 b7 5d fc c0 87 56 4a 31 e2 0d 09 7f dc 1c 52 61 6e 31 49 30 ...5.O..c5...]...VJ1......Ran1I0
08a0 8e 1c 3f 7a b6 e8 2f 01 06 00 45 ac 71 3d 69 56 90 5e 00 00 00 00 49 45 4e 44 ae 42 60 82 ..?z../...E.q=iV.^....IEND.B`.
2'>462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523
/*
 Derived from source code of TrueCrypt 7.1a, which is
 Copyright (c) 2008-2012 TrueCrypt Developers Association and which is governed
 by the TrueCrypt License 3.0.

 Modifications and additions to the original source code (contained in this file)
 and all other portions of this file are Copyright (c) 2013-2017 IDRIX
 and are governed by the Apache License 2.0 the full text of which is
 contained in the file License.txt included in VeraCrypt binary and source
 code distribution packages.
*/

#include "Platform/Platform.h"
#include "Cipher.h"
#include "Crypto/Aes.h"
#include "Crypto/SerpentFast.h"
#include "Crypto/Twofish.h"
#include "Crypto/Camellia.h"
#include "Crypto/kuznyechik.h"

#ifdef TC_AES_HW_CPU
#	include "Crypto/Aes_hw_cpu.h"
#endif

extern "C" int IsAesHwCpuSupported ()
{
#ifdef TC_AES_HW_CPU
	static bool state = false;
	static bool stateValid = false;

	if (!stateValid)
	{
		state = HasAESNI() ? true : false;
		stateValid = true;
	}
	return state && VeraCrypt::Cipher::IsHwSupportEnabled();
#else
	return false;
#endif
}

namespace VeraCrypt
{
	Cipher::Cipher () : Initialized (false)
	{
	}

	Cipher::~Cipher ()
	{
	}

	void Cipher::DecryptBlock (uint8 *data) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

		Decrypt (data);
	}

	void Cipher::DecryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

		while (blockCount-- > 0)
		{
			Decrypt (data);
			data += GetBlockSize();
		}
	}

	void Cipher::EncryptBlock (uint8 *data) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

		Encrypt (data);
	}

	void Cipher::EncryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

		while (blockCount-- > 0)
		{
			Encrypt (data);
			data += GetBlockSize();
		}
	}

	CipherList Cipher::GetAvailableCiphers ()
	{
		CipherList l;

		l.push_back (shared_ptr <Cipher> (new CipherAES ()));
        #ifndef WOLFCRYPT_BACKEND
		l.push_back (shared_ptr <Cipher> (new CipherSerpent ()));
		l.push_back (shared_ptr <Cipher> (new CipherTwofish ()));
		l.push_back (shared_ptr <Cipher> (new CipherCamellia ()));
		l.push_back (shared_ptr <Cipher> (new CipherKuznyechik ()));
        #endif
		return l;
	}

	void Cipher::SetKey (const ConstBufferPtr &key)
	{
		if (key.Size() != GetKeySize ())
			throw ParameterIncorrect (SRC_POS);

		if (!Initialized)
			ScheduledKey.Allocate (GetScheduledKeySize ());

		SetCipherKey (key);
		Key.CopyFrom (key);
		Initialized = true;
	}

    #ifdef WOLFCRYPT_BACKEND
        void Cipher::SetKeyXTS (const ConstBufferPtr &key)
	{
		if (key.Size() != GetKeySize ())
			throw ParameterIncorrect (SRC_POS);

		if (!Initialized)
			ScheduledKey.Allocate (GetScheduledKeySize ());

		SetCipherKeyXTS (key);
		Key.CopyFrom (key);
		Initialized = true;
	}

         void Cipher::EncryptBlockXTS (uint8 *data, uint64 length, uint64 startDataUnitNo) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

		EncryptXTS (data, length, startDataUnitNo);
	}

        void Cipher::DecryptBlockXTS (uint8 *data, uint64 length, uint64 startDataUnitNo) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

		DecryptXTS (data, length, startDataUnitNo);
	}
    #endif

#define TC_EXCEPTION(TYPE) TC_SERIALIZER_FACTORY_ADD(TYPE)
#undef TC_EXCEPTION_NODECL
#define TC_EXCEPTION_NODECL(TYPE) TC_SERIALIZER_FACTORY_ADD(TYPE)

	TC_SERIALIZER_FACTORY_ADD_EXCEPTION_SET (CipherException);


	// AES
	void CipherAES::Decrypt (uint8 *data) const
	{
#ifdef TC_AES_HW_CPU
		if (IsHwSupportAvailable())
			aes_hw_cpu_decrypt (ScheduledKey.Ptr() + sizeof (aes_encrypt_ctx), data);
		else
#endif
			aes_decrypt (data, data, (aes_decrypt_ctx *) (ScheduledKey.Ptr() + sizeof (aes_encrypt_ctx)));
	}

	void CipherAES::DecryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#ifdef TC_AES_HW_CPU
		if ((blockCount & (32 - 1)) == 0
			&& IsHwSupportAvailable())
		{
			while (blockCount > 0)
			{
				aes_hw_cpu_decrypt_32_blocks (ScheduledKey.Ptr() + sizeof (aes_encrypt_ctx), data);

				data += 32 * GetBlockSize();
				blockCount -= 32;
			}
		}
		else
#endif
			Cipher::DecryptBlocks (data, blockCount);
	}

	void CipherAES::Encrypt (uint8 *data) const
	{
#ifdef TC_AES_HW_CPU
		if (IsHwSupportAvailable())
			aes_hw_cpu_encrypt (ScheduledKey.Ptr(), data);
		else
#endif
			aes_encrypt (data, data, (aes_encrypt_ctx *) ScheduledKey.Ptr());
	}

	void CipherAES::EncryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#ifdef TC_AES_HW_CPU
		if ((blockCount & (32 - 1)) == 0
			&& IsHwSupportAvailable())
		{
			while (blockCount > 0)
			{
				aes_hw_cpu_encrypt_32_blocks (ScheduledKey.Ptr(), data);

				data += 32 * GetBlockSize();
				blockCount -= 32;
			}
		}
		else
#endif
			Cipher::EncryptBlocks (data, blockCount);
	}
    #ifdef WOLFCRYPT_BACKEND
        void CipherAES::EncryptXTS (uint8 *data, uint64 length, uint64 startDataUnitNo) const
	{
	    xts_encrypt (data, data, length, startDataUnitNo, (aes_encrypt_ctx *) ScheduledKey.Ptr());
	}

        void CipherAES::DecryptXTS (uint8 *data, uint64 length, uint64 startDataUnitNo) const
	{
	    xts_decrypt (data, data, length, startDataUnitNo, (aes_decrypt_ctx *) (ScheduledKey.Ptr() + sizeof (aes_encrypt_ctx)));
	}

        void CipherAES::SetCipherKeyXTS (const uint8 *key)
	{
		if (xts_encrypt_key256 (key, (aes_encrypt_ctx *) ScheduledKey.Ptr()) != EXIT_SUCCESS)
			throw CipherInitError (SRC_POS);

		if (xts_decrypt_key256 (key, (aes_decrypt_ctx *) (ScheduledKey.Ptr() + sizeof (aes_encrypt_ctx))) != EXIT_SUCCESS)
			throw CipherInitError (SRC_POS);
	}
    #endif

	size_t CipherAES::GetScheduledKeySize () const
	{
		return sizeof(aes_encrypt_ctx) + sizeof(aes_decrypt_ctx);
	}

	bool CipherAES::IsHwSupportAvailable () const
	{
#ifdef TC_AES_HW_CPU
		static bool state = false;
		static bool stateValid = false;

		if (!stateValid)
		{
			state = HasAESNI() ? true : false;
			stateValid = true;
		}
		return state && HwSupportEnabled;
#else
		return false;
#endif
	}

	void CipherAES::SetCipherKey (const uint8 *key)
	{
		if (aes_encrypt_key256 (key, (aes_encrypt_ctx *) ScheduledKey.Ptr()) != EXIT_SUCCESS)
			throw CipherInitError (SRC_POS);

		if (aes_decrypt_key256 (key, (aes_decrypt_ctx *) (ScheduledKey.Ptr() + sizeof (aes_encrypt_ctx))) != EXIT_SUCCESS)
			throw CipherInitError (SRC_POS);
	}

    #ifndef WOLFCRYPT_BACKEND
	// Serpent
	void CipherSerpent::Decrypt (uint8 *data) const
	{
		serpent_decrypt (data, data, ScheduledKey);
	}

	void CipherSerpent::Encrypt (uint8 *data) const
	{
		serpent_encrypt (data, data, ScheduledKey);
	}

	size_t CipherSerpent::GetScheduledKeySize () const
	{
		return 140*4;
	}

	void CipherSerpent::SetCipherKey (const uint8 *key)
	{
		serpent_set_key (key, ScheduledKey);
	}
	
	void CipherSerpent::EncryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#if CRYPTOPP_BOOL_SSE2_INTRINSICS_AVAILABLE && !defined(CRYPTOPP_DISABLE_ASM)
		if ((blockCount >= 4)
			&& IsHwSupportAvailable())
		{
			serpent_encrypt_blocks (data, data, blockCount, ScheduledKey.Ptr());
		}
		else
#endif
			Cipher::EncryptBlocks (data, blockCount);
	}
	
	void CipherSerpent::DecryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#if CRYPTOPP_BOOL_SSE2_INTRINSICS_AVAILABLE && !defined(CRYPTOPP_DISABLE_ASM)
		if ((blockCount >= 4)
			&& IsHwSupportAvailable())
		{
			serpent_decrypt_blocks (data, data, blockCount, ScheduledKey.Ptr());
		}
		else
#endif
			Cipher::DecryptBlocks (data, blockCount);
	}
	
	bool CipherSerpent::IsHwSupportAvailable () const
	{
#if CRYPTOPP_BOOL_SSE2_INTRINSICS_AVAILABLE
		static bool state = false;
		static bool stateValid = false;

		if (!stateValid)
		{
			state = HasSSE2() ? true : false;
			stateValid = true;
		}
		return state;
#else
		return false;
#endif
	}


	// Twofish
	void CipherTwofish::Decrypt (uint8 *data) const
	{
		twofish_decrypt ((TwofishInstance *) ScheduledKey.Ptr(), (unsigned int *)data, (unsigned int *)data);
	}

	void CipherTwofish::Encrypt (uint8 *data) const
	{
		twofish_encrypt ((TwofishInstance *) ScheduledKey.Ptr(), (unsigned int *)data, (unsigned int *)data);
	}

	size_t CipherTwofish::GetScheduledKeySize () const
	{
		return TWOFISH_KS;
	}

	void CipherTwofish::SetCipherKey (const uint8 *key)
	{
		twofish_set_key ((TwofishInstance *) ScheduledKey.Ptr(), (unsigned int *) key);
	}
	
	void CipherTwofish::EncryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#if CRYPTOPP_BOOL_X64 && !defined(CRYPTOPP_DISABLE_ASM)
		twofish_encrypt_blocks ( (TwofishInstance *) ScheduledKey.Ptr(), data, data, blockCount);
#else
		Cipher::EncryptBlocks (data, blockCount);
#endif
	}
	
	void CipherTwofish::DecryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#if CRYPTOPP_BOOL_X64 && !defined(CRYPTOPP_DISABLE_ASM)
		twofish_decrypt_blocks ( (TwofishInstance *) ScheduledKey.Ptr(), data, data, blockCount);
#else
		Cipher::DecryptBlocks (data, blockCount);
#endif
	}
	
	bool CipherTwofish::IsHwSupportAvailable () const
	{
#if CRYPTOPP_BOOL_X64 && !defined(CRYPTOPP_DISABLE_ASM)
		return true;
#else
		return false;
#endif
	}
	
	// Camellia
	void CipherCamellia::Decrypt (uint8 *data) const
	{
		camellia_decrypt (data, data, ScheduledKey.Ptr());
	}

	void CipherCamellia::Encrypt (uint8 *data) const
	{
		camellia_encrypt (data, data, ScheduledKey.Ptr());
	}

	size_t CipherCamellia::GetScheduledKeySize () const
	{
		return CAMELLIA_KS;
	}

	void CipherCamellia::SetCipherKey (const uint8 *key)
	{
		camellia_set_key (key, ScheduledKey.Ptr());
	}
	
	void CipherCamellia::EncryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#if CRYPTOPP_BOOL_X64 && !defined(CRYPTOPP_DISABLE_ASM)
		camellia_encrypt_blocks ( ScheduledKey.Ptr(), data, data, blockCount);
#else
		Cipher::EncryptBlocks (data, blockCount);
#endif
	}
	
	void CipherCamellia::DecryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#if CRYPTOPP_BOOL_X64 && !defined(CRYPTOPP_DISABLE_ASM)
		camellia_decrypt_blocks ( ScheduledKey.Ptr(), data, data, blockCount);
#else
		Cipher::DecryptBlocks (data, blockCount);
#endif
	}
	
	bool CipherCamellia::IsHwSupportAvailable () const
	{
#if CRYPTOPP_BOOL_X64 && !defined(CRYPTOPP_DISABLE_ASM)
		return true;
#else
		return false;
#endif
	}

	// Kuznyechik
	void CipherKuznyechik::Decrypt (uint8 *data) const
	{
		kuznyechik_decrypt_block (data, data, (kuznyechik_kds *) ScheduledKey.Ptr());
	}

	void CipherKuznyechik::Encrypt (uint8 *data) const
	{
		kuznyechik_encrypt_block (data, data, (kuznyechik_kds *) ScheduledKey.Ptr());
	}

	size_t CipherKuznyechik::GetScheduledKeySize () const
	{
		return KUZNYECHIK_KS;
	}

	void CipherKuznyechik::SetCipherKey (const uint8 *key)
	{
		kuznyechik_set_key (key, (kuznyechik_kds *) ScheduledKey.Ptr());
	}
	void CipherKuznyechik::EncryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#if CRYPTOPP_BOOL_SSE2_INTRINSICS_AVAILABLE
		if ((blockCount >= 4)
			&& IsHwSupportAvailable())
		{
			kuznyechik_encrypt_blocks (data, data, blockCount, (kuznyechik_kds *) ScheduledKey.Ptr());
		}
		else
#endif
			Cipher::EncryptBlocks (data, blockCount);
	}
	
	void CipherKuznyechik::DecryptBlocks (uint8 *data, size_t blockCount) const
	{
		if (!Initialized)
			throw NotInitialized (SRC_POS);

#if CRYPTOPP_BOOL_SSE2_INTRINSICS_AVAILABLE
		if ((blockCount >= 4)
			&& IsHwSupportAvailable())
		{
			kuznyechik_decrypt_blocks (data, data, blockCount, (kuznyechik_kds *) ScheduledKey.Ptr());
		}
		else
#endif
			Cipher::DecryptBlocks (data, blockCount);
	}
	
	bool CipherKuznyechik::IsHwSupportAvailable () const
	{
#if CRYPTOPP_BOOL_SSE2_INTRINSICS_AVAILABLE
		static bool state = false;
		static bool stateValid = false;

		if (!stateValid)
		{
			state = HasSSE2() ? true : false;
			stateValid = true;
		}
		return state;
#else
		return false;
#endif
	}
    #endif	
        bool Cipher::HwSupportEnabled = true;
}