blob: 37aa730fe55d969dff696658f32505f1361f3044 [file] [log] [blame]
// Copyright 2022 The Bazel Authors. All rights reserved.
//
// 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.
package com.google.devtools.build.lib.vfs;
import static com.google.common.base.Preconditions.checkState;
import java.io.IOException;
import java.util.Collection;
import javax.annotation.Nullable;
/**
* Centralized point to perform filesystem calls, to promote caching. Ideally all filesystem
* operations would be cached in Skyframe, but even then, implementations of this interface may do
* batch operations and prefetching to improve performance.
*
* <p>There is typically one {@link SyscallCache} instance in effect for the lifetime of the Bazel
* server, set in {@link com.google.devtools.build.lib.runtime.WorkspaceBuilder}. Between commands,
* {@link #clear} is called to drop cached data from the previous command.
*
* <p>See the note in {@link XattrProvider} about caching in implementations. Do not call the
* methods in this interface on files that may change <em>during</em> a build, like outputs or
* external repository files. Calling these methods on source files is allowed.
*/
public interface SyscallCache extends XattrProvider {
SyscallCache NO_CACHE =
new SyscallCache() {
@Override
public Collection<Dirent> readdir(Path path) throws IOException {
return path.readdir(Symlinks.NOFOLLOW);
}
@Nullable
@Override
public FileStatus statIfFound(Path path, Symlinks symlinks) throws IOException {
return path.statIfFound(symlinks);
}
@Override
public DirentTypeWithSkip getType(Path path, Symlinks symlinks) {
return DirentTypeWithSkip.FILESYSTEM_OP_SKIPPED;
}
@Override
public void clear() {}
};
/** Gets directory entries and their types. Does not follow symlinks. */
Collection<Dirent> readdir(Path path) throws IOException;
/** Returns the stat() for the given path, or null. */
@Nullable
FileStatus statIfFound(Path path, Symlinks symlinks) throws IOException;
/**
* Returns the type of a specific file. This may be answered using stat() or readdir(). Returns
* null if the path does not exist. Returns {@link DirentTypeWithSkip#FILESYSTEM_OP_SKIPPED} if
* cache had no data for path and chose not to do filesystem access to determine the type. Callers
* should call {@link #statIfFound} and then {@link #statusToDirentType} if needed in that case.
*/
@Nullable
DirentTypeWithSkip getType(Path path, Symlinks symlinks) throws IOException;
/** Called before each build. Implementations should flush their caches at that point. */
void clear();
/**
* Called at the end of the analysis phase (if not doing merged analysis/execution). Cache may
* choose to drop some data then.
*/
default void noteAnalysisPhaseEnded() {
clear();
}
/**
* A {@link Dirent.Type} with an additional element signifying that the type is unknown because
* this {@link SyscallCache} implementation skipped filesystem access.
*/
enum DirentTypeWithSkip {
FILE(Dirent.Type.FILE),
DIRECTORY(Dirent.Type.DIRECTORY),
SYMLINK(Dirent.Type.SYMLINK),
UNKNOWN(Dirent.Type.UNKNOWN),
FILESYSTEM_OP_SKIPPED(null);
@Nullable private final Dirent.Type type;
DirentTypeWithSkip(@Nullable Dirent.Type type) {
this.type = type;
}
public Dirent.Type getType() {
checkState(this != FILESYSTEM_OP_SKIPPED, "No type if filesystem op skipped");
return type;
}
@Nullable
public static DirentTypeWithSkip of(@Nullable Dirent.Type type) {
if (type == null) {
return null;
}
switch (type) {
case FILE:
return FILE;
case DIRECTORY:
return DIRECTORY;
case SYMLINK:
return SYMLINK;
case UNKNOWN:
return UNKNOWN;
}
throw new IllegalStateException("Got unrecognized type " + type);
}
}
@Nullable
static Dirent.Type statusToDirentType(FileStatus status) {
if (status == null) {
return null;
} else if (status.isFile()) {
return Dirent.Type.FILE;
} else if (status.isDirectory()) {
return Dirent.Type.DIRECTORY;
} else if (status.isSymbolicLink()) {
return Dirent.Type.SYMLINK;
}
return Dirent.Type.UNKNOWN;
}
}