Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Properly support for Windows long file paths through Unicode Win32 API calls. #2410

Open
wants to merge 12 commits into
base: master
Choose a base branch
from
Open
36 changes: 17 additions & 19 deletions src/build_log.cc
Original file line number Diff line number Diff line change
Expand Up @@ -21,13 +21,13 @@
#endif

#include "build_log.h"
#include "disk_interface.h"

#include <cassert>
#include <errno.h>
#include <stdlib.h>
#include <string.h>

#include <cassert>

#ifndef _WIN32
#include <inttypes.h>
#include <unistd.h>
Expand Down Expand Up @@ -72,8 +72,8 @@ BuildLog::LogEntry::LogEntry(const string& output, uint64_t command_hash,
start_time(start_time), end_time(end_time), mtime(mtime)
{}

BuildLog::BuildLog()
: log_file_(NULL), needs_recompaction_(false) {}
BuildLog::BuildLog(DiskInterface& disk_interface)
: disk_interface_(&disk_interface) {}

BuildLog::~BuildLog() {
Close();
Expand Down Expand Up @@ -137,7 +137,7 @@ bool BuildLog::OpenForWriteIfNeeded() {
if (log_file_ || log_file_path_.empty()) {
return true;
}
log_file_ = fopen(log_file_path_.c_str(), "ab");
log_file_ = disk_interface_->OpenFile(log_file_path_, "ab");
if (!log_file_) {
return false;
}
Expand Down Expand Up @@ -211,7 +211,7 @@ struct LineReader {

LoadStatus BuildLog::Load(const string& path, string* err) {
METRIC_RECORD(".ninja_log load");
FILE* file = fopen(path.c_str(), "r");
FILE* file = disk_interface_->OpenFile(path, "r");
if (!file) {
if (errno == ENOENT)
return LOAD_NOT_FOUND;
Expand Down Expand Up @@ -241,7 +241,7 @@ LoadStatus BuildLog::Load(const string& path, string* err) {
}
if (invalid_log_version) {
fclose(file);
unlink(path.c_str());
disk_interface_->RemoveFile(path);
// Don't report this as a failure. A missing build log will cause
// us to rebuild the outputs anyway.
return LOAD_NOT_FOUND;
Expand Down Expand Up @@ -345,8 +345,8 @@ bool BuildLog::Recompact(const string& path, const BuildLogUser& user,
METRIC_RECORD(".ninja_log recompact");

Close();
string temp_path = path + ".recompact";
FILE* f = fopen(temp_path.c_str(), "wb");
std::string temp_path = path + ".recompact";
FILE* f = disk_interface_->OpenFile(temp_path, "wb");
if (!f) {
*err = strerror(errno);
return false;
Expand Down Expand Up @@ -376,28 +376,26 @@ bool BuildLog::Recompact(const string& path, const BuildLogUser& user,
entries_.erase(dead_outputs[i]);

fclose(f);
if (unlink(path.c_str()) < 0) {
if (disk_interface_->RemoveFile(path) < 0) {
*err = strerror(errno);
return false;
}

if (rename(temp_path.c_str(), path.c_str()) < 0) {
if (!disk_interface_->RenameFile(temp_path, path)) {
*err = strerror(errno);
return false;
}

return true;
}

bool BuildLog::Restat(const StringPiece path,
const DiskInterface& disk_interface,
const int output_count, char** outputs,
std::string* const err) {
bool BuildLog::Restat(const StringPiece path, const int output_count,
char** outputs, std::string* const err) {
METRIC_RECORD(".ninja_log restat");

Close();
std::string temp_path = path.AsString() + ".restat";
FILE* f = fopen(temp_path.c_str(), "wb");
FILE* f = disk_interface_->OpenFile(temp_path, "wb");
if (!f) {
*err = strerror(errno);
return false;
Expand All @@ -417,7 +415,7 @@ bool BuildLog::Restat(const StringPiece path,
}
}
if (!skip) {
const TimeStamp mtime = disk_interface.Stat(i->second->output, err);
const TimeStamp mtime = disk_interface_->Stat(i->second->output, err);
if (mtime == -1) {
fclose(f);
return false;
Expand All @@ -433,12 +431,12 @@ bool BuildLog::Restat(const StringPiece path,
}

fclose(f);
if (unlink(path.str_) < 0) {
if (disk_interface_->RemoveFile(path.str_) < 0) {
*err = strerror(errno);
return false;
}

if (rename(temp_path.c_str(), path.str_) < 0) {
if (!disk_interface_->RenameFile(temp_path, path.AsString())) {
*err = strerror(errno);
return false;
}
Expand Down
22 changes: 16 additions & 6 deletions src/build_log.h
Original file line number Diff line number Diff line change
Expand Up @@ -15,9 +15,12 @@
#ifndef NINJA_BUILD_LOG_H_
#define NINJA_BUILD_LOG_H_

#include <string>
#include <stdio.h>

#include <memory>
#include <string>

#include "disk_interface.h"
#include "hash_map.h"
#include "load_status.h"
#include "timestamp.h"
Expand All @@ -41,7 +44,10 @@ struct BuildLogUser {
/// 2) timing information, perhaps for generating reports
/// 3) restat information
struct BuildLog {
BuildLog();
/// Constructor takes a reference to an existing DiskInterface instance.
BuildLog(DiskInterface& disk_interface);

/// Destructor.
~BuildLog();

/// Prepares writing to the log file without actually opening it - that will
Expand Down Expand Up @@ -87,21 +93,25 @@ struct BuildLog {
std::string* err);

/// Restat all outputs in the log
bool Restat(StringPiece path, const DiskInterface& disk_interface,
int output_count, char** outputs, std::string* err);
bool Restat(StringPiece path, int output_count, char** outputs,
std::string* err);

typedef ExternalStringHashMap<LogEntry*>::Type Entries;
const Entries& entries() const { return entries_; }

private:
/// Default destructor is private and never implemented.
BuildLog() = delete;

/// Should be called before using log_file_. When false is returned, errno
/// will be set.
bool OpenForWriteIfNeeded();

DiskInterface* disk_interface_ = nullptr;
Entries entries_;
FILE* log_file_;
FILE* log_file_ = nullptr;
std::string log_file_path_;
bool needs_recompaction_;
bool needs_recompaction_ = false;
};

#endif // NINJA_BUILD_LOG_H_
8 changes: 5 additions & 3 deletions src/build_log_perftest.cc
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,8 @@ struct NoDeadPaths : public BuildLogUser {
};

bool WriteTestData(string* err) {
BuildLog log;
SystemDiskInterface disk_interface;
BuildLog log(disk_interface);

NoDeadPaths no_dead_paths;
if (!log.OpenForWrite(kTestFilename, no_dead_paths, err))
Expand Down Expand Up @@ -109,9 +110,10 @@ int main() {
return 1;
}

SystemDiskInterface disk_interface;
{
// Read once to warm up disk cache.
BuildLog log;
BuildLog log(disk_interface);
if (log.Load(kTestFilename, &err) == LOAD_ERROR) {
fprintf(stderr, "Failed to read test data: %s\n", err.c_str());
return 1;
Expand All @@ -120,7 +122,7 @@ int main() {
const int kNumRepetitions = 5;
for (int i = 0; i < kNumRepetitions; ++i) {
int64_t start = GetTimeMillis();
BuildLog log;
BuildLog log(disk_interface);
if (log.Load(kTestFilename, &err) == LOAD_ERROR) {
fprintf(stderr, "Failed to read test data: %s\n", err.c_str());
return 1;
Expand Down
Loading
Loading