VeraCrypt

Documentation >> Technical Details >> VeraCrypt Volume Format Specification

VeraCrypt Volume Format Specification

The format of file-hosted volumes is identical to the format of partition/device-hosted volumes (however, the "volume header", or key data, for a system partition/drive is stored in the last 512 bytes of the first logical drive track). VeraCrypt volumes have no "signature" or ID strings. Until decrypted, they appear to consist solely of random data.
Free space on each VeraCrypt volume is filled with random data when the volume is created.* The random data is generated as follows: Right before VeraCrypt volume formatting begins, a temporary encryption key and a temporary secondary key (XTS mode) are generated by the random number generator (see the section Random Number Generator). The encryption algorithm that the user selected is initialized with the temporary keys. The encryption algorithm is then used to encrypt plaintext blocks consisting of random bytes generated by the random number generator. The encryption algorithm operates in XTS mode (see the section Hidden Volume). The resulting ciphertext blocks are used to fill (overwrite) the free space on the volume. The temporary keys are stored in RAM and are erased after formatting finishes.


VeraCrypt Volume Format Specification:

Offset (bytes) Size (bytes) Encryption
Status†
Description
       
64 
 Unencrypted§
 Salt
       
64 
 Encrypted
 ASCII string "VERA"
       
68 
 Encrypted
 Volume header format version (2)
       
70 
 Encrypted
 Minimum program version required to open the volume
       
72 
 Encrypted
 CRC-32 checksum of the (decrypted) bytes 256-511
       
76 
16 
 Encrypted
 Reserved (must contain zeroes)
       
92 
 Encrypted
 Size of hidden volume (set to zero in non-hidden volumes)
       
100 
 Encrypted
 Size of volume
       
108 
 Encrypted
 Byte offset of the start of the master key scope
       
116 
 Encrypted
 Size of the encrypted area within the master key scope
       
124 
 Encrypted
 Flag bits (bit 0 set: system encryption; bit 1 set: non-system
 
 
 
 in-place-encrypted/decrypted volume; bits 2–31 are reserved)
       
128 
 Encrypted
 Sector size (in bytes)
       
132 
120 
 Encrypted
 Reserved (must contain zeroes)
       
252 
 Encrypted
 CRC-32 checksum of the (decrypted) bytes 64-251
       
256 
Var. 
 Encrypted
 Concatenated primary and secondary master keys**
       
512 
65024 
 Encrypted
 Reserved (for system encryption, this item is omitted‡‡)
       
65536 
65536 
 Encrypted /
 Area for hidden volume header (if there is no hidden volume
 
 
 Unencrypted§
 within the volume, this area contains random data††). For
 
 
 
 system encryption, this item is omitted.‡‡ See bytes 0–65535.
       
131072 
Var. 
 Encrypted
 Data area (master key scope). For system encryption, offset
 
 
 
 may be different (depending on offset of system partition).
       
S-131072‡
65536 
 Encrypted /
 Backup header (encrypted with a different header key derived
 
 
 Unencrypted§
 using a different salt). For system encryption, this item is
 
 
 
 omitted.‡‡ See bytes 0–65535.
       
S-65536‡
65536 
 Encrypted /
 Backup header for hidden volume (encrypted with a different
 
 
 Unencrypted§
 header key derived using a different salt). If there is no hidden
 
 
 
 volume within the volume, this area contains random data.††
 
 
 
 For system encryption, this item is omitted.‡‡ See bytes
 
 
 
 0–65535.
       
The fields located at byte #0 (salt) and #256 (master keys) contain random values generated by the random number generator (see the section Random Number Generator) during the volume creation process.
If a VeraCrypt volume hosts a hidden volume (within its free space), the header of the hidden volume is located at byte #65536 of the host volume (the header of the host/outer volume is located at byte #0 of the host volume – see the section Hidden Volume). If there is no hidden volume within a VeraCrypt volume, bytes 65536–131071 of the volume (i.e., the area where the header of a hidden volume can reside) contain random data (see above for information on the method used to fill free volume space with random data when the volume is created). The layout of the header of a hidden volume is the same as the one of a standard volume (bytes 0–65535).
The maximum possible VeraCrypt volume size is 263 bytes (8,589,934,592 GB). However, due to security reasons (with respect to the 128-bit block size used by the encryption algorithms), the maximum allowed volume size is 1 PB (1,048,576 GB).

Embedded Backup Headers

Each VeraCrypt volume contains an embedded backup header, located at the end of the volume (see above). The header backup is not a copy of the volume header because it is encrypted with a different header key derived using a different salt (see the section Header Key Derivation, Salt, and Iteration Count).
When the volume password and/or PIM and/or keyfiles are changed, or when the header is restored from the embedded (or an external) header backup, both the volume header and the backup header (embedded in the volume) are re-encrypted with different header keys (derived using newly generated salts – the salt for the volume header is different from the salt for the backup header). Each salt is generated by the VeraCrypt random number generator (see the section Random Number Generator).
For more information about header backups, see the subsection Tools > Restore Volume Header in the chapter Main Program Window.
Next Section >>

* Provided that the options Quick Format and Dynamic are disabled and provided that the volume does not contain a filesystem that has been encrypted in place (note that VeraCrypt does not allow the user to create a hidden volume within such a volume).
† The encrypted areas of the volume header are encrypted in XTS mode using the primary and secondary header keys. For more information, see the section Encryption Scheme and the section Header Key Derivation, Salt, and Iteration Count.
S denotes the size of the volume host (in bytes).
§ Note that the salt does not need to be encrypted, as it does not have to be kept secret [7] (salt is a sequence of random values).
** Multiple concatenated master keys are stored here when the volume is encrypted using a cascade of ciphers (secondary master keys are used for XTS mode).
†† See above in this section for information on the method used to fill free volume space with random data when the volume is created.
‡‡ Here, the meaning of "system encryption" does not include a hidden volume containing a hidden operating system.

ndId::ListVolumes; param1IsMountedVolumeSpec = true; } if (parser.Found (L"list-token-keyfiles")) { CheckCommandSingle(); ArgCommand = CommandId::ListSecurityTokenKeyfiles; } if (parser.Found (L"mount")) { CheckCommandSingle(); ArgCommand = CommandId::MountVolume; param1IsVolume = true; } if (parser.Found (L"save-preferences")) { CheckCommandSingle(); ArgCommand = CommandId::SavePreferences; } if (parser.Found (L"test")) { CheckCommandSingle(); ArgCommand = CommandId::Test; } if (parser.Found (L"volume-properties")) { CheckCommandSingle(); ArgCommand = CommandId::DisplayVolumeProperties; param1IsMountedVolumeSpec = true; } // Options if (parser.Found (L"background-task")) StartBackgroundTask = true; #ifdef TC_WINDOWS if (parser.Found (L"cache")) ArgMountOptions.CachePassword = true; #endif ArgDisplayPassword = parser.Found (L"display-password"); if (parser.Found (L"encryption", &str)) { ArgEncryptionAlgorithm.reset(); foreach (shared_ptr <EncryptionAlgorithm> ea, EncryptionAlgorithm::GetAvailableAlgorithms()) { if (!ea->IsDeprecated() && wxString (ea->GetName()).IsSameAs (str, false)) ArgEncryptionAlgorithm = ea; } if (!ArgEncryptionAlgorithm) throw_err (LangString["UNKNOWN_OPTION"] + L": " + str); } if (parser.Found (L"explore")) Preferences.OpenExplorerWindowAfterMount = true; if (parser.Found (L"filesystem", &str)) { if (str.IsSameAs (L"none", false)) { ArgMountOptions.NoFilesystem = true; ArgFilesystem = VolumeCreationOptions::FilesystemType::None; } else { ArgMountOptions.FilesystemType = wstring (str); if (str.IsSameAs (L"FAT", false)) ArgFilesystem = VolumeCreationOptions::FilesystemType::FAT; #ifdef TC_LINUX else if (str.IsSameAs (L"Ext2", false)) ArgFilesystem = VolumeCreationOptions::FilesystemType::Ext2; else if (str.IsSameAs (L"Ext3", false)) ArgFilesystem = VolumeCreationOptions::FilesystemType::Ext3; else if (str.IsSameAs (L"Ext4", false)) ArgFilesystem = VolumeCreationOptions::FilesystemType::Ext4; else if (str.IsSameAs (L"NTFS", false)) ArgFilesystem = VolumeCreationOptions::FilesystemType::NTFS; else if (str.IsSameAs (L"exFAT", false)) ArgFilesystem = VolumeCreationOptions::FilesystemType::exFAT; #elif defined (TC_MACOSX) else if ( str.IsSameAs (L"HFS", false) || str.IsSameAs (L"HFS+", false) || str.IsSameAs (L"MacOsExt", false) ) { ArgFilesystem = VolumeCreationOptions::FilesystemType::MacOsExt; } else if (str.IsSameAs (L"exFAT", false)) ArgFilesystem = VolumeCreationOptions::FilesystemType::exFAT; #elif defined (TC_FREEBSD) || defined (TC_SOLARIS) else if (str.IsSameAs (L"UFS", false)) ArgFilesystem = VolumeCreationOptions::FilesystemType::UFS; #endif else ArgFilesystem = VolumeCreationOptions::FilesystemType::None; } } ArgForce = parser.Found (L"force"); ArgTrueCryptMode = parser.Found (L"truecrypt"); #if !defined(TC_WINDOWS) && !defined(TC_MACOSX) if (parser.Found (L"fs-options", &str)) ArgMountOptions.FilesystemOptions = str; #endif if (parser.Found (L"hash", &str)) { ArgHash.reset(); foreach (shared_ptr <Hash> hash, Hash::GetAvailableAlgorithms()) { wxString hashName (hash->GetName()); wxString hashAltName (hash->GetAltName()); if (hashName.IsSameAs (str, false) || hashAltName.IsSameAs (str, false)) ArgHash = hash; } if (!ArgHash) throw_err (LangString["UNKNOWN_OPTION"] + L": " + str); } if (parser.Found (L"new-hash", &str)) { ArgNewHash.reset(); foreach (shared_ptr <Hash> hash, Hash::GetAvailableAlgorithms()) { wxString hashName (hash->GetName()); wxString hashAltName (hash->GetAltName()); if (hashName.IsSameAs (str, false) || hashAltName.IsSameAs (str, false)) ArgNewHash = hash; } if (!ArgNewHash) throw_err (LangString["UNKNOWN_OPTION"] + L": " + str); } if (parser.Found (L"keyfiles", &str)) ArgKeyfiles = ToKeyfileList (str); if (parser.Found (L"mount-options", &str)) { wxStringTokenizer tokenizer (str, L","); while (tokenizer.HasMoreTokens()) { wxString token = tokenizer.GetNextToken(); if (token == L"headerbak") ArgMountOptions.UseBackupHeaders = true; else if (token == L"nokernelcrypto") ArgMountOptions.NoKernelCrypto = true; else if (token == L"readonly" || token == L"ro") ArgMountOptions.Protection = VolumeProtection::ReadOnly; else if (token == L"system") ArgMountOptions.PartitionInSystemEncryptionScope = true; else if (token == L"timestamp" || token == L"ts") ArgMountOptions.PreserveTimestamps = false; #ifdef TC_WINDOWS else if (token == L"removable" || token == L"rm") ArgMountOptions.Removable = true; #endif else throw_err (LangString["UNKNOWN_OPTION"] + L": " + token); } } if (parser.Found (L"new-keyfiles", &str)) ArgNewKeyfiles = ToKeyfileList (str); if (parser.Found (L"new-password", &str)) ArgNewPassword = ToUTF8Password (str.c_str()); if (parser.Found (L"new-pim", &str)) { try { ArgNewPim = StringConverter::ToInt32 (wstring (str)); } catch (...) { throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); } if (ArgNewPim < 0 || ArgNewPim > (ArgMountOptions.PartitionInSystemEncryptionScope? MAX_BOOT_PIM_VALUE: MAX_PIM_VALUE)) throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); else if (ArgNewPim > 0 && ArgTrueCryptMode) throw_err (LangString["PIM_NOT_SUPPORTED_FOR_TRUECRYPT_MODE"]); } if (parser.Found (L"non-interactive")) { if (interfaceType != UserInterfaceType::Text) throw_err (L"--non-interactive is supported only in text mode"); Preferences.NonInteractive = true; } if (parser.Found (L"stdin")) { if (!Preferences.NonInteractive) throw_err (L"--stdin is supported only in non-interactive mode"); Preferences.UseStandardInput = true; } if (parser.Found (L"password", &str)) { if (Preferences.UseStandardInput) throw_err (L"--password cannot be used with --stdin"); ArgPassword = ToUTF8Password (str.c_str()); } if (parser.Found (L"pim", &str)) { try { ArgPim = StringConverter::ToInt32 (wstring (str)); } catch (...) { throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); } if (ArgPim < 0 || ArgPim > (ArgMountOptions.PartitionInSystemEncryptionScope? MAX_BOOT_PIM_VALUE: MAX_PIM_VALUE)) throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); else if (ArgPim > 0 && ArgTrueCryptMode) throw_err (LangString["PIM_NOT_SUPPORTED_FOR_TRUECRYPT_MODE"]); } if (parser.Found (L"protect-hidden", &str)) { if (str == L"yes") { if (ArgMountOptions.Protection != VolumeProtection::ReadOnly) ArgMountOptions.Protection = VolumeProtection::HiddenVolumeReadOnly; } else if (str == L"no") ArgNoHiddenVolumeProtection = true; else throw_err (LangString["UNKNOWN_OPTION"] + L": " + str); } if (parser.Found (L"protection-keyfiles", &str)) { ArgMountOptions.ProtectionKeyfiles = ToKeyfileList (str); ArgMountOptions.Protection = VolumeProtection::HiddenVolumeReadOnly; } if (parser.Found (L"protection-password", &str)) { ArgMountOptions.ProtectionPassword = ToUTF8Password (str.c_str()); ArgMountOptions.Protection = VolumeProtection::HiddenVolumeReadOnly; } if (parser.Found (L"protection-pim", &str)) { int pim = -1; try { pim = StringConverter::ToInt32 (wstring (str)); if (pim < 0 || pim > (ArgMountOptions.PartitionInSystemEncryptionScope? MAX_BOOT_PIM_VALUE: MAX_PIM_VALUE)) throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); } catch (...) { throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); } ArgMountOptions.ProtectionPim = pim; ArgMountOptions.Protection = VolumeProtection::HiddenVolumeReadOnly; } if (parser.Found (L"protection-hash", &str)) { bool bHashFound = false; foreach (shared_ptr <Hash> hash, Hash::GetAvailableAlgorithms()) { wxString hashName (hash->GetName()); wxString hashAltName (hash->GetAltName()); if (hashName.IsSameAs (str, false) || hashAltName.IsSameAs (str, false)) { bHashFound = true; ArgMountOptions.ProtectionKdf = Pkcs5Kdf::GetAlgorithm (*hash, ArgTrueCryptMode); } } if (!bHashFound) throw_err (LangString["UNKNOWN_OPTION"] + L": " + str); } ArgQuick = parser.Found (L"quick"); if (parser.Found (L"random-source", &str)) ArgRandomSourcePath = FilesystemPath (str.wc_str()); if (parser.Found (L"restore-headers")) { CheckCommandSingle(); ArgCommand = CommandId::RestoreHeaders; param1IsVolume = true; } if (parser.Found (L"slot", &str)) { unsigned long number; if (!str.ToULong (&number) || number < Core->GetFirstSlotNumber() || number > Core->GetLastSlotNumber()) throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); ArgMountOptions.SlotNumber = number; if (param1IsMountedVolumeSpec) { shared_ptr <VolumeInfo> volume = Core->GetMountedVolume (number); if (!volume) throw_err (_("No such volume is mounted.")); ArgVolumes.push_back (volume); param1IsMountedVolumeSpec = false; } } if (parser.Found (L"size", &str)) { uint64 multiplier; wxChar lastChar = str [str.Length () - 1]; if (lastChar >= wxT('0') && lastChar <= wxT('9')) multiplier = 1; else if (lastChar == wxT('K') || lastChar == wxT('k')) multiplier = BYTES_PER_KB; else if (lastChar == wxT('M') || lastChar == wxT('m')) multiplier = BYTES_PER_MB; else if (lastChar == wxT('G') || lastChar == wxT('g')) multiplier = BYTES_PER_GB; else if (lastChar == wxT('T') || lastChar == wxT('t')) multiplier = BYTES_PER_TB; else throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); // remove suffix if present if (multiplier != 1) str.RemoveLast (); // check that we only have digits in the string size_t index = str.find_first_not_of (wxT("0123456789")); if (index != (size_t) wxNOT_FOUND) { // restore last characater for error display if (multiplier != 1) str += lastChar; throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); } try { ArgSize = multiplier * StringConverter::ToUInt64 (wstring (str)); } catch (...) { throw_err (LangString["PARAMETER_INCORRECT"] + L": " + str); } } if (parser.Found (L"token-lib", &str)) Preferences.SecurityTokenModule = wstring (str); if (parser.Found (L"token-pin", &str) && !str.IsEmpty ()) { ArgTokenPin = ToUTF8Buffer (str.c_str(), str.Len ()); } if (parser.Found (L"verbose")) Preferences.Verbose = true; if (parser.Found (L"volume-type", &str)) { if (str.IsSameAs (L"normal", false)) ArgVolumeType = VolumeType::Normal; else if (str.IsSameAs (L"hidden", false)) ArgVolumeType = VolumeType::Hidden; else throw_err (LangString["UNKNOWN_OPTION"] + L": " + str); } // Parameters if (parser.GetParamCount() > 0) { // in case of GUI interface, we load the preference when only // specifying volume path without any option/switch if (Application::GetUserInterfaceType() != UserInterfaceType::Text) { // check if only parameters were specified in the command line // (e.g. when associating .hc extension in mimetype with /usr/bin/veracrypt) bool onlyParametersPresent = (parser.GetParamCount() == (size_t) (argc - 1)); if (onlyParametersPresent) { // no options/switches, so we load prefences now Preferences.Load(); ArgMountOptions = Preferences.DefaultMountOptions; } } if (ArgCommand == CommandId::None) { ArgCommand = CommandId::MountVolume; param1IsVolume = true; } if (param1IsVolume) { wxFileName volPath (parser.GetParam (0)); #ifdef TC_WINDOWS if (!parser.GetParam (0).StartsWith (L"\\Device\\")) #endif volPath.Normalize (wxPATH_NORM_ABSOLUTE | wxPATH_NORM_DOTS); ArgVolumePath.reset (new VolumePath (wstring (volPath.GetFullPath()))); } if (param1IsMountPoint || parser.GetParamCount() >= 2) { wstring s (parser.GetParam (param1IsMountPoint ? 0 : 1)); if (s.empty()) ArgMountOptions.NoFilesystem = true; wxFileName mountPoint (wstring (Directory::AppendSeparator (s))); mountPoint.Normalize (wxPATH_NORM_ABSOLUTE | wxPATH_NORM_DOTS); ArgMountPoint.reset (new DirectoryPath (wstring (mountPoint.GetPath()))); } if (param1IsFile) { ArgFilePath.reset (new FilePath (parser.GetParam (0).wc_str())); } } if (param1IsMountedVolumeSpec) ArgVolumes = GetMountedVolumes (parser.GetParamCount() > 0 ? parser.GetParam (0) : wxString()); if (ArgCommand == CommandId::None && Application::GetUserInterfaceType() == UserInterfaceType::Text) parser.Usage(); } CommandLineInterface::~CommandLineInterface () { } void CommandLineInterface::CheckCommandSingle () const { if (ArgCommand != CommandId::None) throw_err (_("Only a single command can be specified at a time.")); } shared_ptr <KeyfileList> CommandLineInterface::ToKeyfileList (const wxString &arg) const { wxStringTokenizer tokenizer (arg, L",", wxTOKEN_RET_EMPTY_ALL); // Handle escaped separator wxArrayString arr; bool prevEmpty = false; while (tokenizer.HasMoreTokens()) { wxString token = tokenizer.GetNextToken(); if (prevEmpty && token.empty() && tokenizer.HasMoreTokens()) { token = tokenizer.GetNextToken(); if (!token.empty()) { arr.Add (token); prevEmpty = true; continue; } } if (token.empty() && !tokenizer.HasMoreTokens()) break; if (prevEmpty || token.empty()) { if (arr.Count() < 1) { arr.Add (L""); continue; } arr.Last() += token.empty() ? L"," : token.wc_str(); } else arr.Add (token); prevEmpty = token.empty(); } make_shared_auto (KeyfileList, keyfileList); for (size_t i = 0; i < arr.GetCount(); i++) { if (!arr[i].empty()) keyfileList->push_back (make_shared <Keyfile> (wstring (arr[i]))); } return keyfileList; } VolumeInfoList CommandLineInterface::GetMountedVolumes (const wxString &mountedVolumeSpec) const { VolumeInfoList volumes = Core->GetMountedVolumes (); VolumeInfoList filteredVolumes; wxFileName pathFilter; if (!mountedVolumeSpec.empty()) { pathFilter = mountedVolumeSpec; pathFilter.Normalize (wxPATH_NORM_ABSOLUTE | wxPATH_NORM_DOTS); } else return volumes; foreach (shared_ptr <VolumeInfo> volume, volumes) { if (mountedVolumeSpec.empty()) { filteredVolumes.push_back (volume); } else if (wxString (wstring(volume->Path)) == pathFilter.GetFullPath()) { filteredVolumes.push_back (volume); } else if (wxString (wstring(volume->MountPoint)) == pathFilter.GetFullPath() || (wxString (wstring(volume->MountPoint)) + wxFileName::GetPathSeparator()) == pathFilter.GetFullPath()) { filteredVolumes.push_back (volume); } } if (!mountedVolumeSpec.IsEmpty() && filteredVolumes.size() < 1) throw_err (_("No such volume is mounted.")); return filteredVolumes; } shared_ptr<VolumePassword> ToUTF8Password (const wchar_t* str, size_t charCount) { if (charCount > 0) { shared_ptr<SecureBuffer> utf8Buffer = ToUTF8Buffer (str, charCount); return shared_ptr<VolumePassword>(new VolumePassword (*utf8Buffer)); } else return shared_ptr<VolumePassword>(new VolumePassword ()); } shared_ptr<SecureBuffer> ToUTF8Buffer (const wchar_t* str, size_t charCount) { if (charCount == (size_t) -1) charCount = wcslen (str); if (charCount > 0) { wxMBConvUTF8 utf8; size_t ulen = utf8.FromWChar (NULL, 0, str, charCount); if (wxCONV_FAILED == ulen) throw PasswordUTF8Invalid (SRC_POS); SecureBuffer passwordBuf(ulen); ulen = utf8.FromWChar ((char*) (byte*) passwordBuf, ulen, str, charCount); if (wxCONV_FAILED == ulen) throw PasswordUTF8Invalid (SRC_POS); if (ulen > VolumePassword::MaxSize) throw PasswordUTF8TooLong (SRC_POS); ConstBufferPtr utf8Buffer ((byte*) passwordBuf, ulen); return shared_ptr<SecureBuffer>(new SecureBuffer (utf8Buffer)); } else return shared_ptr<SecureBuffer>(new SecureBuffer ()); } auto_ptr <CommandLineInterface> CmdLine; }