@ -12,9 +12,11 @@
# include <cctype>
# include <iostream>
# include "env/composite_env_wrapper.h"
# include "env/env_encryption_ctr.h"
# include "monitoring/perf_context_imp.h"
# include "rocksdb/convenience.h"
# include "rocksdb/io_status.h"
# include "rocksdb/system_clock.h"
# include "util/aligned_buffer.h"
# include "util/coding.h"
@ -85,19 +87,24 @@ std::shared_ptr<EncryptionProvider> EncryptionProvider::NewCTRProvider(
// If an error was encountered, returns a non-OK status.
//
// REQUIRES: External synchronization
Status EncryptedSequentialFile : : Read ( size_t n , Slice * result , char * scratch ) {
IOStatus EncryptedSequentialFile : : Read ( size_t n , const IOOptions & options ,
Slice * result , char * scratch ,
IODebugContext * dbg ) {
assert ( scratch ) ;
Status status = file_ - > Read ( n , result , scratch ) ;
if ( ! statu s. ok ( ) ) {
return statu s;
IOStatus io_ s = file_ - > Read ( n , options , result , scratch , dbg ) ;
if ( ! io_ s. ok ( ) ) {
return io_ s;
}
{
PERF_TIMER_GUARD ( decrypt_data_nanos ) ;
status = stream_ - > Decrypt ( offset_ , ( char * ) result - > data ( ) , result - > size ( ) ) ;
io_s = status_to_io_status (
stream_ - > Decrypt ( offset_ , ( char * ) result - > data ( ) , result - > size ( ) ) ) ;
}
if ( io_s . ok ( ) ) {
offset_ + = result - > size ( ) ; // We've already ready data from disk, so update
// offset_ even if decryption fails.
return status ;
}
return io_s ;
}
// Skip "n" bytes from the file. This is guaranteed to be no
@ -107,7 +114,7 @@ Status EncryptedSequentialFile::Read(size_t n, Slice* result, char* scratch) {
// file, and Skip will return OK.
//
// REQUIRES: External synchronization
Status EncryptedSequentialFile : : Skip ( uint64_t n ) {
IO Status EncryptedSequentialFile : : Skip ( uint64_t n ) {
auto status = file_ - > Skip ( n ) ;
if ( ! status . ok ( ) ) {
return status ;
@ -131,26 +138,30 @@ size_t EncryptedSequentialFile::GetRequiredBufferAlignment() const {
// Remove any kind of caching of data from the offset to offset+length
// of this file. If the length is 0, then it refers to the end of file.
// If the system is not caching the file contents, then this is a noop.
Status EncryptedSequentialFile : : InvalidateCache ( size_t offset , size_t length ) {
IOStatus EncryptedSequentialFile : : InvalidateCache ( size_t offset ,
size_t length ) {
return file_ - > InvalidateCache ( offset + prefixLength_ , length ) ;
}
// Positioned Read for direct I/O
// If Direct I/O enabled, offset, n, and scratch should be properly aligned
Status EncryptedSequentialFile : : PositionedRead ( uint64_t offset , size_t n ,
Slice * result , char * scratch ) {
IOStatus EncryptedSequentialFile : : PositionedRead ( uint64_t offset , size_t n ,
const IOOptions & options ,
Slice * result , char * scratch ,
IODebugContext * dbg ) {
assert ( scratch ) ;
offset + = prefixLength_ ; // Skip prefix
auto statu s = file_ - > PositionedRead ( offset , n , result , scratch ) ;
if ( ! statu s. ok ( ) ) {
return statu s;
auto io_ s = file_ - > PositionedRead ( offset , n , options , result , scratch , dbg ) ;
if ( ! io_ s. ok ( ) ) {
return io_ s;
}
offset_ = offset + result - > size ( ) ;
{
PERF_TIMER_GUARD ( decrypt_data_nanos ) ;
status = stream_ - > Decrypt ( offset , ( char * ) result - > data ( ) , result - > size ( ) ) ;
io_s = status_to_io_status (
stream_ - > Decrypt ( offset , ( char * ) result - > data ( ) , result - > size ( ) ) ) ;
}
return statu s;
return io_ s;
}
// Read up to "n" bytes from the file starting at "offset".
@ -163,25 +174,30 @@ Status EncryptedSequentialFile::PositionedRead(uint64_t offset, size_t n,
//
// Safe for concurrent use by multiple threads.
// If Direct I/O enabled, offset, n, and scratch should be aligned properly.
Status EncryptedRandomAccessFile : : Read ( uint64_t offset , size_t n , Slice * result ,
char * scratch ) const {
IOStatus EncryptedRandomAccessFile : : Read ( uint64_t offset , size_t n ,
const IOOptions & options ,
Slice * result , char * scratch ,
IODebugContext * dbg ) const {
assert ( scratch ) ;
offset + = prefixLength_ ;
auto statu s = file_ - > Read ( offset , n , result , scratch ) ;
if ( ! statu s. ok ( ) ) {
return statu s;
auto io_ s = file_ - > Read ( offset , n , options , result , scratch , dbg ) ;
if ( ! io_ s. ok ( ) ) {
return io_ s;
}
{
PERF_TIMER_GUARD ( decrypt_data_nanos ) ;
status = stream_ - > Decrypt ( offset , ( char * ) result - > data ( ) , result - > size ( ) ) ;
io_s = status_to_io_status (
stream_ - > Decrypt ( offset , ( char * ) result - > data ( ) , result - > size ( ) ) ) ;
}
return statu s;
return io_ s;
}
// Readahead the file starting from offset by n bytes for caching.
Status EncryptedRandomAccessFile : : Prefetch ( uint64_t offset , size_t n ) {
IOStatus EncryptedRandomAccessFile : : Prefetch ( uint64_t offset , size_t n ,
const IOOptions & options ,
IODebugContext * dbg ) {
// return Status::OK();
return file_ - > Prefetch ( offset + prefixLength_ , n ) ;
return file_ - > Prefetch ( offset + prefixLength_ , n , options , dbg ) ;
}
// Tries to get an unique ID for this file that will be the same each time
@ -222,7 +238,7 @@ size_t EncryptedRandomAccessFile::GetRequiredBufferAlignment() const {
// Remove any kind of caching of data from the offset to offset+length
// of this file. If the length is 0, then it refers to the end of file.
// If the system is not caching the file contents, then this is a noop.
Status EncryptedRandomAccessFile : : InvalidateCache ( size_t offset ,
IO Status EncryptedRandomAccessFile : : InvalidateCache ( size_t offset ,
size_t length ) {
return file_ - > InvalidateCache ( offset + prefixLength_ , length ) ;
}
@ -230,12 +246,13 @@ Status EncryptedRandomAccessFile::InvalidateCache(size_t offset,
// A file abstraction for sequential writing. The implementation
// must provide buffering since callers may append small fragments
// at a time to the file.
Status EncryptedWritableFile : : Append ( const Slice & data ) {
IOStatus EncryptedWritableFile : : Append ( const Slice & data ,
const IOOptions & options ,
IODebugContext * dbg ) {
AlignedBuffer buf ;
Status status ;
Slice dataToAppend ( data ) ;
if ( data . size ( ) > 0 ) {
auto offset = file_ - > GetFileSize ( ) ; // size including prefix
auto offset = file_ - > GetFileSize ( options , dbg ) ; // size including prefix
// Encrypt in cloned buffer
buf . Alignment ( GetRequiredBufferAlignment ( ) ) ;
buf . AllocateNewBuffer ( data . size ( ) ) ;
@ -243,26 +260,25 @@ Status EncryptedWritableFile::Append(const Slice& data) {
// so that the next two lines can be replaced with buf.Append().
memmove ( buf . BufferStart ( ) , data . data ( ) , data . size ( ) ) ;
buf . Size ( data . size ( ) ) ;
IOStatus io_s ;
{
PERF_TIMER_GUARD ( encrypt_data_nanos ) ;
status = stream_ - > Encrypt ( offset , buf . BufferStart ( ) , buf . CurrentSize ( ) ) ;
io_s = status_to_io_status (
stream_ - > Encrypt ( offset , buf . BufferStart ( ) , buf . CurrentSize ( ) ) ) ;
}
if ( ! statu s. ok ( ) ) {
return statu s;
if ( ! io_ s. ok ( ) ) {
return io_ s;
}
dataToAppend = Slice ( buf . BufferStart ( ) , buf . CurrentSize ( ) ) ;
}
status = file_ - > Append ( dataToAppend ) ;
if ( ! status . ok ( ) ) {
return status ;
}
return status ;
return file_ - > Append ( dataToAppend , options , dbg ) ;
}
Status EncryptedWritableFile : : PositionedAppend ( const Slice & data ,
uint64_t offset ) {
IOStatus EncryptedWritableFile : : PositionedAppend ( const Slice & data ,
uint64_t offset ,
const IOOptions & options ,
IODebugContext * dbg ) {
AlignedBuffer buf ;
Status status ;
Slice dataToAppend ( data ) ;
offset + = prefixLength_ ;
if ( data . size ( ) > 0 ) {
@ -271,20 +287,18 @@ Status EncryptedWritableFile::PositionedAppend(const Slice& data,
buf . AllocateNewBuffer ( data . size ( ) ) ;
memmove ( buf . BufferStart ( ) , data . data ( ) , data . size ( ) ) ;
buf . Size ( data . size ( ) ) ;
IOStatus io_s ;
{
PERF_TIMER_GUARD ( encrypt_data_nanos ) ;
status = stream_ - > Encrypt ( offset , buf . BufferStart ( ) , buf . CurrentSize ( ) ) ;
io_s = status_to_io_status (
stream_ - > Encrypt ( offset , buf . BufferStart ( ) , buf . CurrentSize ( ) ) ) ;
}
if ( ! statu s. ok ( ) ) {
return statu s;
if ( ! io_ s. ok ( ) ) {
return io_ s;
}
dataToAppend = Slice ( buf . BufferStart ( ) , buf . CurrentSize ( ) ) ;
}
status = file_ - > PositionedAppend ( dataToAppend , offset ) ;
if ( ! status . ok ( ) ) {
return status ;
}
return status ;
return file_ - > PositionedAppend ( dataToAppend , offset , options , dbg ) ;
}
// Indicates the upper layers if the current WritableFile implementation
@ -302,48 +316,72 @@ size_t EncryptedWritableFile::GetRequiredBufferAlignment() const {
/*
* Get the size of valid data in the file .
*/
uint64_t EncryptedWritableFile : : GetFileSize ( ) {
return file_ - > GetFileSize ( ) - prefixLength_ ;
uint64_t EncryptedWritableFile : : GetFileSize ( const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > GetFileSize ( options , dbg ) - prefixLength_ ;
}
// Truncate is necessary to trim the file to the correct size
// before closing. It is not always possible to keep track of the file
// size due to whole pages writes. The behavior is undefined if called
// with other writes to follow.
Status EncryptedWritableFile : : Truncate ( uint64_t size ) {
return file_ - > Truncate ( size + prefixLength_ ) ;
// Truncate is necessary to trim the file to the correct size
// before closing. It is not always possible to keep track of the file
// size due to whole pages writes. The behavior is undefined if called
// with other writes to follow.
IOStatus EncryptedWritableFile : : Truncate ( uint64_t size ,
const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > Truncate ( size + prefixLength_ , options , dbg ) ;
}
// Remove any kind of caching of data from the offset to offset+length
// of this file. If the length is 0, then it refers to the end of file.
// If the system is not caching the file contents, then this is a noop.
// This call has no effect on dirty pages in the cache.
Status EncryptedWritableFile : : InvalidateCache ( size_t offset , size_t length ) {
// Remove any kind of caching of data from the offset to offset+length
// of this file. If the length is 0, then it refers to the end of file.
// If the system is not caching the file contents, then this is a noop.
// This call has no effect on dirty pages in the cache.
IO Status EncryptedWritableFile : : InvalidateCache ( size_t offset , size_t length ) {
return file_ - > InvalidateCache ( offset + prefixLength_ , length ) ;
}
// Sync a file range with disk.
// offset is the starting byte of the file range to be synchronized.
// nbytes specifies the length of the range to be synchronized.
// This asks the OS to initiate flushing the cached data to disk,
// without waiting for completion.
// Default implementation does nothing.
Status EncryptedWritableFile : : RangeSync ( uint64_t offset , uint64_t nbytes ) {
return file_ - > RangeSync ( offset + prefixLength_ , nbytes ) ;
// Sync a file range with disk.
// offset is the starting byte of the file range to be synchronized.
// nbytes specifies the length of the range to be synchronized.
// This asks the OS to initiate flushing the cached data to disk,
// without waiting for completion.
// Default implementation does nothing.
IOStatus EncryptedWritableFile : : RangeSync ( uint64_t offset , uint64_t nbytes ,
const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > RangeSync ( offset + prefixLength_ , nbytes , options , dbg ) ;
}
// PrepareWrite performs any necessary preparation for a write
// before the write actually occurs. This allows for pre-allocation
// of space on devices where it can result in less file
// fragmentation and/or less waste from over-zealous filesystem
// pre-allocation.
void EncryptedWritableFile : : PrepareWrite ( size_t offset , size_t len ) {
file_ - > PrepareWrite ( offset + prefixLength_ , len ) ;
// PrepareWrite performs any necessary preparation for a write
// before the write actually occurs. This allows for pre-allocation
// of space on devices where it can result in less file
// fragmentation and/or less waste from over-zealous filesystem
// pre-allocation.
void EncryptedWritableFile : : PrepareWrite ( size_t offset , size_t len ,
const IOOptions & options ,
IODebugContext * dbg ) {
file_ - > PrepareWrite ( offset + prefixLength_ , len , options , dbg ) ;
}
// Pre-allocates space for a file.
Status EncryptedWritableFile : : Allocate ( uint64_t offset , uint64_t len ) {
return file_ - > Allocate ( offset + prefixLength_ , len ) ;
// Pre-allocates space for a file.
IOStatus EncryptedWritableFile : : Allocate ( uint64_t offset , uint64_t len ,
const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > Allocate ( offset + prefixLength_ , len , options , dbg ) ;
}
IOStatus EncryptedWritableFile : : Flush ( const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > Flush ( options , dbg ) ;
}
IOStatus EncryptedWritableFile : : Sync ( const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > Sync ( options , dbg ) ;
}
IOStatus EncryptedWritableFile : : Close ( const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > Close ( options , dbg ) ;
}
// A file abstraction for random reading and writing.
@ -362,9 +400,10 @@ size_t EncryptedRandomRWFile::GetRequiredBufferAlignment() const {
// Write bytes in `data` at offset `offset`, Returns Status::OK() on success.
// Pass aligned buffer when use_direct_io() returns true.
Status EncryptedRandomRWFile : : Write ( uint64_t offset , const Slice & data ) {
IOStatus EncryptedRandomRWFile : : Write ( uint64_t offset , const Slice & data ,
const IOOptions & options ,
IODebugContext * dbg ) {
AlignedBuffer buf ;
Status status ;
Slice dataToWrite ( data ) ;
offset + = prefixLength_ ;
if ( data . size ( ) > 0 ) {
@ -373,71 +412,89 @@ Status EncryptedRandomRWFile::Write(uint64_t offset, const Slice& data) {
buf . AllocateNewBuffer ( data . size ( ) ) ;
memmove ( buf . BufferStart ( ) , data . data ( ) , data . size ( ) ) ;
buf . Size ( data . size ( ) ) ;
IOStatus io_s ;
{
PERF_TIMER_GUARD ( encrypt_data_nanos ) ;
status = stream_ - > Encrypt ( offset , buf . BufferStart ( ) , buf . CurrentSize ( ) ) ;
io_s = status_to_io_status (
stream_ - > Encrypt ( offset , buf . BufferStart ( ) , buf . CurrentSize ( ) ) ) ;
}
if ( ! statu s. ok ( ) ) {
return statu s;
if ( ! io_ s. ok ( ) ) {
return io_ s;
}
dataToWrite = Slice ( buf . BufferStart ( ) , buf . CurrentSize ( ) ) ;
}
status = file_ - > Write ( offset , dataToWrite ) ;
return status ;
return file_ - > Write ( offset , dataToWrite , options , dbg ) ;
}
// Read up to `n` bytes starting from offset `offset` and store them in
// result, provided `scratch` size should be at least `n`.
// Returns Status::OK() on success.
Status EncryptedRandomRWFile : : Read ( uint64_t offset , size_t n , Slice * result ,
char * scratch ) const {
IOStatus EncryptedRandomRWFile : : Read ( uint64_t offset , size_t n ,
const IOOptions & options , Slice * result ,
char * scratch , IODebugContext * dbg ) const {
assert ( scratch ) ;
offset + = prefixLength_ ;
auto status = file_ - > Read ( offset , n , result , scratch ) ;
auto status = file_ - > Read ( offset , n , options , result , scratch , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
{
PERF_TIMER_GUARD ( decrypt_data_nanos ) ;
status = stream_ - > Decrypt ( offset , ( char * ) result - > data ( ) , result - > size ( ) ) ;
status = status_to_io_status (
stream_ - > Decrypt ( offset , ( char * ) result - > data ( ) , result - > size ( ) ) ) ;
}
return status ;
}
Status EncryptedRandomRWFile : : Flush ( ) { return file_ - > Flush ( ) ; }
IOStatus EncryptedRandomRWFile : : Flush ( const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > Flush ( options , dbg ) ;
}
Status EncryptedRandomRWFile : : Sync ( ) { return file_ - > Sync ( ) ; }
IOStatus EncryptedRandomRWFile : : Sync ( const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > Sync ( options , dbg ) ;
}
Status EncryptedRandomRWFile : : Fsync ( ) { return file_ - > Fsync ( ) ; }
IOStatus EncryptedRandomRWFile : : Fsync ( const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > Fsync ( options , dbg ) ;
}
Status EncryptedRandomRWFile : : Close ( ) { return file_ - > Close ( ) ; }
IOStatus EncryptedRandomRWFile : : Close ( const IOOptions & options ,
IODebugContext * dbg ) {
return file_ - > Close ( options , dbg ) ;
}
// EncryptedEnv implements an Env wrapper that adds encryption to files stored
// on disk.
class EncryptedEnvImpl : public EnvWrapper {
namespace {
// EncryptedFileSystemImpl implements an FileSystemWrapper that adds encryption
// to files stored on disk.
class EncryptedFileSystemImpl : public EncryptedFileSystem {
public :
const char * Name ( ) const override { return " EncryptedFS " ; }
// Returns the raw encryption provider that should be used to write the input
// encrypted file. If there is no such provider, NotFound is returned.
Status GetWritableProvider ( const std : : string & /*fname*/ ,
IO Status GetWritableProvider ( const std : : string & /*fname*/ ,
EncryptionProvider * * result ) {
if ( provider_ ) {
* result = provider_ . get ( ) ;
return Status : : OK ( ) ;
return IO Status: : OK ( ) ;
} else {
* result = nullptr ;
return Status : : NotFound ( " No WriteProvider specified " ) ;
return IO Status: : NotFound ( " No WriteProvider specified " ) ;
}
}
// Returns the raw encryption provider that should be used to read the input
// encrypted file. If there is no such provider, NotFound is returned.
Status GetReadableProvider ( const std : : string & /*fname*/ ,
IO Status GetReadableProvider ( const std : : string & /*fname*/ ,
EncryptionProvider * * result ) {
if ( provider_ ) {
* result = provider_ . get ( ) ;
return Status : : OK ( ) ;
return IO Status: : OK ( ) ;
} else {
* result = nullptr ;
return Status : : NotFound ( " No Provider specified " ) ;
return IO Status: : NotFound ( " No Provider specified " ) ;
}
}
@ -453,13 +510,13 @@ class EncryptedEnvImpl : public EnvWrapper {
// should be encrypted
// @return OK on success, non-OK on failure.
template < class TypeFile >
Status CreateWritableCipherStream (
IO Status CreateWritableCipherStream (
const std : : string & fname , const std : : unique_ptr < TypeFile > & underlying ,
const Env Options& options , size_t * prefix_length ,
std : : unique_ptr < BlockAccessCipherStream > * stream ) {
const File Options& options , size_t * prefix_length ,
std : : unique_ptr < BlockAccessCipherStream > * stream , IODebugContext * dbg ) {
EncryptionProvider * provider = nullptr ;
* prefix_length = 0 ;
Status status = GetWritableProvider ( fname , & provider ) ;
IO Status status = GetWritableProvider ( fname , & provider ) ;
if ( ! status . ok ( ) ) {
return status ;
} else if ( provider ! = nullptr ) {
@ -471,34 +528,36 @@ class EncryptedEnvImpl : public EnvWrapper {
// Initialize prefix
buffer . Alignment ( underlying - > GetRequiredBufferAlignment ( ) ) ;
buffer . AllocateNewBuffer ( * prefix_length ) ;
status = provider - > CreateNewPrefix ( fname , buffer . BufferStart ( ) ,
* prefix_length ) ;
status = status_to_io_status ( provider - > CreateNewPrefix (
fname , buffer . BufferStart ( ) , * prefix_length ) ) ;
if ( status . ok ( ) ) {
buffer . Size ( * prefix_length ) ;
prefix = Slice ( buffer . BufferStart ( ) , buffer . CurrentSize ( ) ) ;
// Write prefix
status = underlying - > Append ( prefix ) ;
status = underlying - > Append ( prefix , options . io_options , dbg ) ;
}
if ( ! status . ok ( ) ) {
return status ;
}
}
// Create cipher stream
status = provider - > CreateCipherStream ( fname , options , prefix , stream ) ;
status = status_to_io_status (
provider - > CreateCipherStream ( fname , options , prefix , stream ) ) ;
}
return status ;
}
template < class TypeFile >
Status CreateWritableEncryptedFile ( const std : : string & fname ,
IO Status CreateWritableEncryptedFile ( const std : : string & fname ,
std : : unique_ptr < TypeFile > & underlying ,
const EnvOptions & options ,
std : : unique_ptr < TypeFile > * result ) {
const FileOptions & options ,
std : : unique_ptr < TypeFile > * result ,
IODebugContext * dbg ) {
// Create cipher stream
std : : unique_ptr < BlockAccessCipherStream > stream ;
size_t prefix_length ;
Status status = CreateWritableCipherStream ( fname , underlying , options ,
& prefix_length , & stream ) ;
IO Status status = CreateWritableCipherStream ( fname , underlying , options ,
& prefix_length , & stream , dbg ) ;
if ( status . ok ( ) ) {
if ( stream ) {
result - > reset ( new EncryptedWritableFile (
@ -522,15 +581,15 @@ class EncryptedEnvImpl : public EnvWrapper {
// should be encrypted
// @return OK on success, non-OK on failure.
template < class TypeFile >
Status CreateRandomWriteCipherStream (
IO Status CreateRandomWriteCipherStream (
const std : : string & fname , const std : : unique_ptr < TypeFile > & underlying ,
const Env Options& options , size_t * prefix_length ,
std : : unique_ptr < BlockAccessCipherStream > * stream ) {
const File Options& options , size_t * prefix_length ,
std : : unique_ptr < BlockAccessCipherStream > * stream , IODebugContext * dbg ) {
EncryptionProvider * provider = nullptr ;
* prefix_length = 0 ;
Status statu s = GetWritableProvider ( fname , & provider ) ;
if ( ! statu s. ok ( ) ) {
return statu s;
IOStatus io_ s = GetWritableProvider ( fname , & provider ) ;
if ( ! io_ s. ok ( ) ) {
return io_ s;
} else if ( provider ! = nullptr ) {
// Initialize & write prefix (if needed)
AlignedBuffer buffer ;
@ -540,22 +599,23 @@ class EncryptedEnvImpl : public EnvWrapper {
// Initialize prefix
buffer . Alignment ( underlying - > GetRequiredBufferAlignment ( ) ) ;
buffer . AllocateNewBuffer ( * prefix_length ) ;
status = provider - > CreateNewPrefix ( fname , buffer . BufferStart ( ) ,
* prefix_length ) ;
if ( statu s. ok ( ) ) {
io_s = status_to_io_status ( provider - > CreateNewPrefix (
fname , buffer . BufferStart ( ) , * prefix_length ) ) ;
if ( io_ s. ok ( ) ) {
buffer . Size ( * prefix_length ) ;
prefix = Slice ( buffer . BufferStart ( ) , buffer . CurrentSize ( ) ) ;
// Write prefix
statu s = underlying - > Write ( 0 , prefix ) ;
io_ s = underlying - > Write ( 0 , prefix , options . io_options , dbg ) ;
}
if ( ! statu s. ok ( ) ) {
return statu s;
if ( ! io_ s. ok ( ) ) {
return io_ s;
}
}
// Create cipher stream
status = provider - > CreateCipherStream ( fname , options , prefix , stream ) ;
io_s = status_to_io_status (
provider - > CreateCipherStream ( fname , options , prefix , stream ) ) ;
}
return statu s;
return io_ s;
}
// Creates a CipherStream for the underlying file/name using the options
@ -570,10 +630,10 @@ class EncryptedEnvImpl : public EnvWrapper {
// is encrypted
// @return OK on success, non-OK on failure.
template < class TypeFile >
Status CreateSequentialCipherStream (
IO Status CreateSequentialCipherStream (
const std : : string & fname , const std : : unique_ptr < TypeFile > & underlying ,
const Env Options& options , size_t * prefix_length ,
std : : unique_ptr < BlockAccessCipherStream > * stream ) {
const File Options& options , size_t * prefix_length ,
std : : unique_ptr < BlockAccessCipherStream > * stream , IODebugContext * dbg ) {
// Read prefix (if needed)
AlignedBuffer buffer ;
Slice prefix ;
@ -582,14 +642,15 @@ class EncryptedEnvImpl : public EnvWrapper {
// Read prefix
buffer . Alignment ( underlying - > GetRequiredBufferAlignment ( ) ) ;
buffer . AllocateNewBuffer ( * prefix_length ) ;
Status status =
underlying - > Read ( * prefix_length , & prefix , buffer . BufferStart ( ) ) ;
IO Status status = underlying - > Read ( * prefix_length , options . io_options ,
& prefix , buffer . BufferStart ( ) , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
buffer . Size ( * prefix_length ) ;
}
return provider_ - > CreateCipherStream ( fname , options , prefix , stream ) ;
return status_to_io_status (
provider_ - > CreateCipherStream ( fname , options , prefix , stream ) ) ;
}
// Creates a CipherStream for the underlying file/name using the options
@ -604,10 +665,10 @@ class EncryptedEnvImpl : public EnvWrapper {
// is encrypted
// @return OK on success, non-OK on failure.
template < class TypeFile >
Status CreateRandomReadCipherStream (
IO Status CreateRandomReadCipherStream (
const std : : string & fname , const std : : unique_ptr < TypeFile > & underlying ,
const Env Options& options , size_t * prefix_length ,
std : : unique_ptr < BlockAccessCipherStream > * stream ) {
const File Options& options , size_t * prefix_length ,
std : : unique_ptr < BlockAccessCipherStream > * stream , IODebugContext * dbg ) {
// Read prefix (if needed)
AlignedBuffer buffer ;
Slice prefix ;
@ -616,39 +677,48 @@ class EncryptedEnvImpl : public EnvWrapper {
// Read prefix
buffer . Alignment ( underlying - > GetRequiredBufferAlignment ( ) ) ;
buffer . AllocateNewBuffer ( * prefix_length ) ;
Status status =
underlying - > Read ( 0 , * prefix_length , & prefix , buffer . BufferStart ( ) ) ;
IO Status status = underlying - > Read ( 0 , * prefix_length , options . io_options ,
& prefix , buffer . BufferStart ( ) , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
buffer . Size ( * prefix_length ) ;
}
return provider_ - > CreateCipherStream ( fname , options , prefix , stream ) ;
return status_to_io_status (
provider_ - > CreateCipherStream ( fname , options , prefix , stream ) ) ;
}
public :
EncryptedEnvImpl ( Env * base_env ,
EncryptedFileSystemImpl ( const std : : shared_ptr < FileSystem > & base ,
const std : : shared_ptr < EncryptionProvider > & provider )
: EnvWrapper ( base_env ) {
: EncryptedFileSystem ( base ) {
provider_ = provider ;
}
Status AddCipher ( const std : : string & descriptor , const char * cipher ,
size_t len , bool for_write ) override {
return provider_ - > AddCipher ( descriptor , cipher , len , for_write ) ;
}
// NewSequentialFile opens a file for sequential reading.
virtual Status NewSequentialFile ( const std : : string & fname ,
std : : unique_ptr < SequentialFile > * result ,
const EnvOptions & options ) override {
IOStatus NewSequentialFile ( const std : : string & fname ,
const FileOptions & options ,
std : : unique_ptr < FSSequentialFile > * result ,
IODebugContext * dbg ) override {
result - > reset ( ) ;
if ( options . use_mmap_reads ) {
return Status : : InvalidArgument ( ) ;
return IO Status: : InvalidArgument ( ) ;
}
// Open file using underlying Env implementation
std : : unique_ptr < SequentialFile > underlying ;
auto status = EnvWrapper : : NewSequentialFile ( fname , & underlying , options ) ;
std : : unique_ptr < FSSequentialFile > underlying ;
auto status =
FileSystemWrapper : : NewSequentialFile ( fname , options , & underlying , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
uint64_t file_size ;
status = EnvWrapper : : GetFileSize ( fname , & file_size ) ;
status = FileSystemWrapper : : GetFileSize ( fname , options . io_options ,
& file_size , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
@ -660,7 +730,7 @@ class EncryptedEnvImpl : public EnvWrapper {
std : : unique_ptr < BlockAccessCipherStream > stream ;
size_t prefix_length ;
status = CreateSequentialCipherStream ( fname , underlying , options ,
& prefix_length , & stream ) ;
& prefix_length , & stream , dbg ) ;
if ( status . ok ( ) ) {
result - > reset ( new EncryptedSequentialFile (
std : : move ( underlying ) , std : : move ( stream ) , prefix_length ) ) ;
@ -669,23 +739,25 @@ class EncryptedEnvImpl : public EnvWrapper {
}
// NewRandomAccessFile opens a file for random read access.
virtual Status NewRandomAccessFile ( const std : : string & fname ,
std : : unique_ptr < RandomAccessFile > * result ,
const EnvOptions & options ) override {
IOStatus NewRandomAccessFile ( const std : : string & fname ,
const FileOptions & options ,
std : : unique_ptr < FSRandomAccessFile > * result ,
IODebugContext * dbg ) override {
result - > reset ( ) ;
if ( options . use_mmap_reads ) {
return Status : : InvalidArgument ( ) ;
return IO Status: : InvalidArgument ( ) ;
}
// Open file using underlying Env implementation
std : : unique_ptr < RandomAccessFile > underlying ;
auto status = EnvWrapper : : NewRandomAccessFile ( fname , & underlying , options ) ;
std : : unique_ptr < FSRandomAccessFile > underlying ;
auto status = FileSystemWrapper : : NewRandomAccessFile ( fname , options ,
& underlying , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
std : : unique_ptr < BlockAccessCipherStream > stream ;
size_t prefix_length ;
status = CreateRandomReadCipherStream ( fname , underlying , options ,
& prefix_length , & stream ) ;
& prefix_length , & stream , dbg ) ;
if ( status . ok ( ) ) {
if ( stream ) {
result - > reset ( new EncryptedRandomAccessFile (
@ -698,20 +770,21 @@ class EncryptedEnvImpl : public EnvWrapper {
}
// NewWritableFile opens a file for sequential writing.
virtual Status NewWritableFile ( const std : : string & fname ,
std : : unique_ptr < WritableFile > * result ,
const EnvOptions & options ) override {
IO Status NewWritableFile ( const std : : string & fname , const FileOptions & options ,
std : : unique_ptr < FS WritableFile> * result ,
IODebugContext * dbg ) override {
result - > reset ( ) ;
if ( options . use_mmap_writes ) {
return Status : : InvalidArgument ( ) ;
return IO Status: : InvalidArgument ( ) ;
}
// Open file using underlying Env implementation
std : : unique_ptr < WritableFile > underlying ;
Status status = EnvWrapper : : NewWritableFile ( fname , & underlying , options ) ;
std : : unique_ptr < FSWritableFile > underlying ;
IOStatus status =
FileSystemWrapper : : NewWritableFile ( fname , options , & underlying , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
return CreateWritableEncryptedFile ( fname , underlying , options , result ) ;
return CreateWritableEncryptedFile ( fname , underlying , options , result , dbg ) ;
}
// Create an object that writes to a new file with the specified
@ -721,39 +794,42 @@ class EncryptedEnvImpl : public EnvWrapper {
// returns non-OK.
//
// The returned file will only be accessed by one thread at a time.
virtual Status ReopenWritableFile ( const std : : string & fname ,
std : : unique_ptr < WritableFile > * result ,
const EnvOptions & options ) override {
IOStatus ReopenWritableFile ( const std : : string & fname ,
const FileOptions & options ,
std : : unique_ptr < FSWritableFile > * result ,
IODebugContext * dbg ) override {
result - > reset ( ) ;
if ( options . use_mmap_writes ) {
return Status : : InvalidArgument ( ) ;
return IO Status: : InvalidArgument ( ) ;
}
// Open file using underlying Env implementation
std : : unique_ptr < WritableFile > underlying ;
Status status = EnvWrapper : : ReopenWritableFile ( fname , & underlying , options ) ;
std : : unique_ptr < FSWritableFile > underlying ;
IOStatus status =
FileSystemWrapper : : ReopenWritableFile ( fname , options , & underlying , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
return CreateWritableEncryptedFile ( fname , underlying , options , result ) ;
return CreateWritableEncryptedFile ( fname , underlying , options , result , dbg ) ;
}
// Reuse an existing file by renaming it and opening it as writable.
virtual Status ReuseWritableFile ( const std : : string & fname ,
IO Status ReuseWritableFile ( const std : : string & fname ,
const std : : string & old_fname ,
std : : unique_ptr < WritableFile > * result ,
const EnvOptions & options ) override {
const FileOptions & options ,
std : : unique_ptr < FSWritableFile > * result ,
IODebugContext * dbg ) override {
result - > reset ( ) ;
if ( options . use_mmap_writes ) {
return Status : : InvalidArgument ( ) ;
return IO Status: : InvalidArgument ( ) ;
}
// Open file using underlying Env implementation
std : : unique_ptr < WritableFile > underlying ;
Status status =
EnvWrapper : : ReuseWritableFile ( fname , old_fname , & underlying , options ) ;
std : : unique_ptr < FS WritableFile> underlying ;
auto status = FileSystemWrapper : : ReuseWritableFile (
fname , old_fname , options , & underlying , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
return CreateWritableEncryptedFile ( fname , underlying , options , result ) ;
return CreateWritableEncryptedFile ( fname , underlying , options , result , dbg ) ;
}
// Open `fname` for random read and write, if file doesn't exist the file
@ -761,19 +837,20 @@ class EncryptedEnvImpl : public EnvWrapper {
// *result and returns OK. On failure returns non-OK.
//
// The returned file will only be accessed by one thread at a time.
virtual Status NewRandomRWFile ( const std : : string & fname ,
std : : unique_ptr < RandomRWFile > * result ,
const EnvOptions & options ) override {
IO Status NewRandomRWFile ( const std : : string & fname , const FileOptions & options ,
std : : unique_ptr < FS RandomRWFile> * result ,
IODebugContext * dbg ) override {
result - > reset ( ) ;
if ( options . use_mmap_reads | | options . use_mmap_writes ) {
return Status : : InvalidArgument ( ) ;
return IO Status: : InvalidArgument ( ) ;
}
// Check file exists
bool isNewFile = ! FileExists ( fname ) . ok ( ) ;
bool isNewFile = ! FileExists ( fname , options . io_options , dbg ) . ok ( ) ;
// Open file using underlying Env implementation
std : : unique_ptr < RandomRWFile > underlying ;
Status status = EnvWrapper : : NewRandomRWFile ( fname , & underlying , options ) ;
std : : unique_ptr < FSRandomRWFile > underlying ;
auto status =
FileSystemWrapper : : NewRandomRWFile ( fname , options , & underlying , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
@ -783,10 +860,10 @@ class EncryptedEnvImpl : public EnvWrapper {
if ( ! isNewFile ) {
// File already exists, read prefix
status = CreateRandomReadCipherStream ( fname , underlying , options ,
& prefix_length , & stream ) ;
& prefix_length , & stream , dbg ) ;
} else {
status = CreateRandomWriteCipherStream ( fname , underlying , options ,
& prefix_length , & stream ) ;
& prefix_length , & stream , dbg ) ;
}
if ( status . ok ( ) ) {
if ( stream ) {
@ -813,9 +890,12 @@ class EncryptedEnvImpl : public EnvWrapper {
// have
// permission to access "dir", or if "dir" is invalid.
// IOError if an IO Error was encountered
virtual Status GetChildrenFileAttributes (
const std : : string & dir , std : : vector < FileAttributes > * result ) override {
auto status = EnvWrapper : : GetChildrenFileAttributes ( dir , result ) ;
IOStatus GetChildrenFileAttributes ( const std : : string & dir ,
const IOOptions & options ,
std : : vector < FileAttributes > * result ,
IODebugContext * dbg ) override {
auto status =
FileSystemWrapper : : GetChildrenFileAttributes ( dir , options , result , dbg ) ;
if ( ! status . ok ( ) ) {
return status ;
}
@ -833,13 +913,14 @@ class EncryptedEnvImpl : public EnvWrapper {
it - > size_bytes - = provider - > GetPrefixLength ( ) ;
}
}
return Status : : OK ( ) ;
return IO Status: : OK ( ) ;
}
// Store the size of fname in *file_size.
virtual Status GetFileSize ( const std : : string & fname ,
uint64_t * file_size ) override {
auto status = EnvWrapper : : GetFileSize ( fname , file_size ) ;
IOStatus GetFileSize ( const std : : string & fname , const IOOptions & options ,
uint64_t * file_size , IODebugContext * dbg ) override {
auto status =
FileSystemWrapper : : GetFileSize ( fname , options , file_size , dbg ) ;
if ( ! status . ok ( ) | | ! ( * file_size ) ) {
return status ;
}
@ -856,12 +937,19 @@ class EncryptedEnvImpl : public EnvWrapper {
private :
std : : shared_ptr < EncryptionProvider > provider_ ;
} ;
} // namespace
std : : shared_ptr < FileSystem > NewEncryptedFS (
const std : : shared_ptr < FileSystem > & base ,
const std : : shared_ptr < EncryptionProvider > & provider ) {
return std : : make_shared < EncryptedFileSystemImpl > ( base , provider ) ;
}
// Returns an Env that encrypts data when stored on disk and decrypts data when
// read from disk.
Env * NewEncryptedEnv ( Env * base_env ,
const std : : shared_ptr < EncryptionProvider > & provider ) {
return new EncryptedEnvImpl ( base_env , provider ) ;
return new CompositeEnvWrapper (
base_env , NewEncryptedFS ( base_env - > GetFileSystem ( ) , provider ) ) ;
}
// Encrypt one or more (partial) blocks of data at the file offset.