| /* |
| * Copyright 2004 The WebRTC Project Authors. All rights reserved. |
| * |
| * Use of this source code is governed by a BSD-style license |
| * that can be found in the LICENSE file in the root of the source |
| * tree. An additional intellectual property rights grant can be found |
| * in the file PATENTS. All contributing project authors may |
| * be found in the AUTHORS file in the root of the source tree. |
| */ |
| |
| #ifndef _WEBRTC_BASE_CRYPTSTRING_H_ |
| #define _WEBRTC_BASE_CRYPTSTRING_H_ |
| |
| #include <string.h> |
| |
| #include <string> |
| #include <vector> |
| |
| #include "webrtc/base/linked_ptr.h" |
| #include "webrtc/base/scoped_ptr.h" |
| |
| namespace rtc { |
| |
| class CryptStringImpl { |
| public: |
| virtual ~CryptStringImpl() {} |
| virtual size_t GetLength() const = 0; |
| virtual void CopyTo(char * dest, bool nullterminate) const = 0; |
| virtual std::string UrlEncode() const = 0; |
| virtual CryptStringImpl * Copy() const = 0; |
| virtual void CopyRawTo(std::vector<unsigned char> * dest) const = 0; |
| }; |
| |
| class EmptyCryptStringImpl : public CryptStringImpl { |
| public: |
| virtual ~EmptyCryptStringImpl() {} |
| virtual size_t GetLength() const { return 0; } |
| virtual void CopyTo(char * dest, bool nullterminate) const { |
| if (nullterminate) { |
| *dest = '\0'; |
| } |
| } |
| virtual std::string UrlEncode() const { return ""; } |
| virtual CryptStringImpl * Copy() const { return new EmptyCryptStringImpl(); } |
| virtual void CopyRawTo(std::vector<unsigned char> * dest) const { |
| dest->clear(); |
| } |
| }; |
| |
| class CryptString { |
| public: |
| CryptString() : impl_(new EmptyCryptStringImpl()) {} |
| size_t GetLength() const { return impl_->GetLength(); } |
| void CopyTo(char * dest, bool nullterminate) const { impl_->CopyTo(dest, nullterminate); } |
| CryptString(const CryptString & other) : impl_(other.impl_->Copy()) {} |
| explicit CryptString(const CryptStringImpl & impl) : impl_(impl.Copy()) {} |
| CryptString & operator=(const CryptString & other) { |
| if (this != &other) { |
| impl_.reset(other.impl_->Copy()); |
| } |
| return *this; |
| } |
| void Clear() { impl_.reset(new EmptyCryptStringImpl()); } |
| std::string UrlEncode() const { return impl_->UrlEncode(); } |
| void CopyRawTo(std::vector<unsigned char> * dest) const { |
| return impl_->CopyRawTo(dest); |
| } |
| |
| private: |
| scoped_ptr<const CryptStringImpl> impl_; |
| }; |
| |
| |
| // Used for constructing strings where a password is involved and we |
| // need to ensure that we zero memory afterwards |
| class FormatCryptString { |
| public: |
| FormatCryptString() { |
| storage_ = new char[32]; |
| capacity_ = 32; |
| length_ = 0; |
| storage_[0] = 0; |
| } |
| |
| void Append(const std::string & text) { |
| Append(text.data(), text.length()); |
| } |
| |
| void Append(const char * data, size_t length) { |
| EnsureStorage(length_ + length + 1); |
| memcpy(storage_ + length_, data, length); |
| length_ += length; |
| storage_[length_] = '\0'; |
| } |
| |
| void Append(const CryptString * password) { |
| size_t len = password->GetLength(); |
| EnsureStorage(length_ + len + 1); |
| password->CopyTo(storage_ + length_, true); |
| length_ += len; |
| } |
| |
| size_t GetLength() { |
| return length_; |
| } |
| |
| const char * GetData() { |
| return storage_; |
| } |
| |
| |
| // Ensures storage of at least n bytes |
| void EnsureStorage(size_t n) { |
| if (capacity_ >= n) { |
| return; |
| } |
| |
| size_t old_capacity = capacity_; |
| char * old_storage = storage_; |
| |
| for (;;) { |
| capacity_ *= 2; |
| if (capacity_ >= n) |
| break; |
| } |
| |
| storage_ = new char[capacity_]; |
| |
| if (old_capacity) { |
| memcpy(storage_, old_storage, length_); |
| |
| // zero memory in a way that an optimizer won't optimize it out |
| old_storage[0] = 0; |
| for (size_t i = 1; i < old_capacity; i++) { |
| old_storage[i] = old_storage[i - 1]; |
| } |
| delete[] old_storage; |
| } |
| } |
| |
| ~FormatCryptString() { |
| if (capacity_) { |
| storage_[0] = 0; |
| for (size_t i = 1; i < capacity_; i++) { |
| storage_[i] = storage_[i - 1]; |
| } |
| } |
| delete[] storage_; |
| } |
| private: |
| char * storage_; |
| size_t capacity_; |
| size_t length_; |
| }; |
| |
| class InsecureCryptStringImpl : public CryptStringImpl { |
| public: |
| std::string& password() { return password_; } |
| const std::string& password() const { return password_; } |
| |
| virtual ~InsecureCryptStringImpl() {} |
| virtual size_t GetLength() const { return password_.size(); } |
| virtual void CopyTo(char * dest, bool nullterminate) const { |
| memcpy(dest, password_.data(), password_.size()); |
| if (nullterminate) dest[password_.size()] = 0; |
| } |
| virtual std::string UrlEncode() const { return password_; } |
| virtual CryptStringImpl * Copy() const { |
| InsecureCryptStringImpl * copy = new InsecureCryptStringImpl; |
| copy->password() = password_; |
| return copy; |
| } |
| virtual void CopyRawTo(std::vector<unsigned char> * dest) const { |
| dest->resize(password_.size()); |
| memcpy(&dest->front(), password_.data(), password_.size()); |
| } |
| private: |
| std::string password_; |
| }; |
| |
| } |
| |
| #endif // _WEBRTC_BASE_CRYPTSTRING_H_ |