Files
swift-mirror/include/swift/Basic/Fingerprint.h
Ben Barham ef8825bfe6 Migrate llvm::Optional to std::optional
LLVM has removed llvm::Optional, move over to std::optional. Also
clang-format to fix up all the renamed #includes.
2024-02-21 11:20:06 -08:00

147 lines
4.9 KiB
C++

//===--- Fingerprint.h - A stable identity for compiler data ----*- C++ -*-===//
//
// This source file is part of the Swift.org open source project
//
// Copyright (c) 2014 - 2020 Apple Inc. and the Swift project authors
// Licensed under Apache License v2.0 with Runtime Library Exception
//
// See https://swift.org/LICENSE.txt for license information
// See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
//
//===----------------------------------------------------------------------===//
#ifndef SWIFT_BASIC_FINGERPRINT_H
#define SWIFT_BASIC_FINGERPRINT_H
#include "swift/Basic/StableHasher.h"
#include "llvm/ADT/Hashing.h"
#include "llvm/ADT/SmallString.h"
#include "llvm/ADT/StringRef.h"
#include "llvm/Support/raw_ostream.h"
#include <optional>
#include <string>
namespace llvm {
namespace yaml {
class IO;
}
} // namespace llvm
namespace swift {
/// A \c Fingerprint represents a stable summary of a given piece of data
/// in the compiler.
///
/// A \c Fingerprint value is subject to the following invariants:
/// 1) For two values \c x and \c y of type T, if \c T::operator==(x, y) is
/// \c true, then the Fingerprint of \c x and the Fingerprint of \c y must be
/// equal.
/// 2) For two values \c x and \c y of type T, the chance of a collision in
/// fingerprints is a rare occurrence - especially if \c y is a minor
/// perturbation of \c x.
/// 3) The \c Fingerprint value is required to be stable *across compilation
/// sessions*.
///
/// Property 3) is the most onerous. It implies that data like addresses, file
/// paths, and other ephemeral compiler state *may not* be used as inputs to the
/// fingerprint generation function.
///
/// \c Fingerprint values are currently used in two places by the compiler's
/// dependency tracking subsystem. They are used at the level of files to detect
/// when tokens (outside of the body of a function or an iterable decl context)
/// have been perturbed. Additionally, they are used at the level of individual
/// iterable decl contexts to detect when the tokens in their bodies have
/// changed. This makes them a coarse - yet safe - overapproximation for when a
/// decl has changed semantically.
class Fingerprint final {
public:
/// The size (in bytes) of the raw value of all fingerprints.
constexpr static size_t DIGEST_LENGTH = 32;
using Core = std::pair<uint64_t, uint64_t>;
private:
Core core;
friend struct StableHasher::Combiner<swift::Fingerprint>;
public:
/// Creates a fingerprint value from a pair of 64-bit integers.
explicit Fingerprint(Fingerprint::Core value) : core(value) {}
/// Creates a fingerprint value from the given input string that is known to
/// be a 32-byte hash value, i.e. that represent a valid 32-bit hex integer.
///
/// Strings that violate this invariant will return a null optional.
static std::optional<Fingerprint> fromString(llvm::StringRef value);
/// Creates a fingerprint value by consuming the given \c StableHasher.
explicit Fingerprint(StableHasher &&stableHasher)
: core{std::move(stableHasher).finalize()} {}
public:
/// Retrieve the raw underlying bytes of this fingerprint.
llvm::SmallString<Fingerprint::DIGEST_LENGTH> getRawValue() const;
public:
friend bool operator==(const Fingerprint &lhs, const Fingerprint &rhs) {
return lhs.core == rhs.core;
}
friend bool operator!=(const Fingerprint &lhs, const Fingerprint &rhs) {
return lhs.core != rhs.core;
}
friend llvm::hash_code hash_value(const Fingerprint &fp) {
return llvm::hash_value(fp.core);
}
public:
bool operator<(const Fingerprint &other) const {
return core < other.core;
}
public:
/// The fingerprint value consisting of 32 bytes of zeroes.
///
/// This fingerprint is a perfectly fine value for a hash, but it is
/// completely arbitrary.
static Fingerprint ZERO() {
return Fingerprint(Fingerprint::Core{0, 0});
}
private:
/// llvm::yaml would like us to be default constructible, but \c Fingerprint
/// would prefer to enforce its internal invariants.
///
/// Very well, LLVM. A default value you shall have.
friend class llvm::yaml::IO;
Fingerprint() : core{Fingerprint::Core{0, 0}} {}
};
void simple_display(llvm::raw_ostream &out, const Fingerprint &fp);
} // namespace swift
namespace swift {
template <> struct StableHasher::Combiner<Fingerprint> {
static void combine(StableHasher &hasher, const Fingerprint &Val) {
// Our underlying buffer is already byte-swapped. Combine the
// raw bytes from the core by hand.
uint8_t buffer[8];
memcpy(buffer, &Val.core.first, sizeof(buffer));
hasher.combine<sizeof(buffer)>(buffer);
memcpy(buffer, &Val.core.second, sizeof(buffer));
hasher.combine<sizeof(buffer)>(buffer);
}
};
} // namespace swift
namespace llvm {
class raw_ostream;
raw_ostream &operator<<(raw_ostream &OS, const swift::Fingerprint &fp);
} // namespace llvm
#endif // SWIFT_BASIC_FINGERPRINT_H