| // |
| // Copyright (C) 2014 The Android Open Source Project |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // http://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| // |
| |
| #include "update_engine/update_manager/real_random_provider.h" |
| |
| #include <stdio.h> |
| #include <unistd.h> |
| |
| #include <string> |
| |
| #include <base/files/file_path.h> |
| #include <base/files/scoped_file.h> |
| #include <base/strings/stringprintf.h> |
| |
| #include "update_engine/update_manager/variable.h" |
| |
| using std::string; |
| |
| namespace { |
| |
| // The device providing randomness. |
| const char* kRandomDevice = "/dev/urandom"; |
| |
| } // namespace |
| |
| namespace chromeos_update_manager { |
| |
| // A random seed variable. |
| class RandomSeedVariable : public Variable<uint64_t> { |
| public: |
| // RandomSeedVariable is initialized as kVariableModeConst to let the |
| // EvaluationContext cache the value between different evaluations of the same |
| // policy request. |
| RandomSeedVariable(const string& name, FILE* fp) |
| : Variable<uint64_t>(name, kVariableModeConst), fp_(fp) {} |
| ~RandomSeedVariable() override {} |
| |
| protected: |
| const uint64_t* GetValue(base::TimeDelta /* timeout */, |
| string* errmsg) override { |
| uint64_t result; |
| // Aliasing via char pointer abides by the C/C++ strict-aliasing rules. |
| char* const buf = reinterpret_cast<char*>(&result); |
| unsigned int buf_rd = 0; |
| |
| while (buf_rd < sizeof(result)) { |
| int rd = fread(buf + buf_rd, 1, sizeof(result) - buf_rd, fp_.get()); |
| if (rd == 0 || ferror(fp_.get())) { |
| // Either EOF on fp or read failed. |
| if (errmsg) { |
| *errmsg = base::StringPrintf( |
| "Error reading from the random device: %s", kRandomDevice); |
| } |
| return nullptr; |
| } |
| buf_rd += rd; |
| } |
| |
| return new uint64_t(result); |
| } |
| |
| private: |
| base::ScopedFILE fp_; |
| |
| DISALLOW_COPY_AND_ASSIGN(RandomSeedVariable); |
| }; |
| |
| bool RealRandomProvider::Init(void) { |
| FILE* fp = fopen(kRandomDevice, "r"); |
| if (!fp) |
| return false; |
| var_seed_.reset(new RandomSeedVariable("seed", fp)); |
| return true; |
| } |
| |
| } // namespace chromeos_update_manager |