blob: ff1685566da869fd76446d08414cdbac970dde9e [file] [log] [blame]
// Copyright 2015 the V8 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.
#ifndef V8_CODEGEN_SIGNATURE_H_
#define V8_CODEGEN_SIGNATURE_H_
#include "src/base/hashing.h"
#include "src/base/vector.h"
#include "src/codegen/machine-type.h"
#include "src/sandbox/check.h"
#include "src/zone/zone.h"
namespace v8 {
namespace internal {
template <typename SigT, typename T>
class SignatureBuilder {
public:
SignatureBuilder(Zone* zone, size_t return_count, size_t parameter_count)
: return_count_(return_count),
parameter_count_(parameter_count),
rcursor_(0),
pcursor_(0) {
// Allocate memory for the signature plus the array backing the
// signature.
constexpr size_t padding = sizeof(SigT) % alignof(T);
using AllocationTypeTag = SignatureBuilder;
const size_t allocated_bytes =
sizeof(SigT) + padding + sizeof(T) * (return_count + parameter_count);
void* memory = zone->Allocate<AllocationTypeTag>(allocated_bytes);
uint8_t* rep_buffer =
reinterpret_cast<uint8_t*>(memory) + sizeof(SigT) + padding;
DCHECK(IsAligned(reinterpret_cast<uintptr_t>(rep_buffer), alignof(T)));
buffer_ = reinterpret_cast<T*>(rep_buffer);
sig_ = new (memory) SigT{return_count, parameter_count, buffer_};
}
const size_t return_count_;
const size_t parameter_count_;
void AddReturn(T val) {
DCHECK_LT(rcursor_, return_count_);
buffer_[rcursor_++] = val;
}
void AddReturnAt(size_t index, T val) {
DCHECK_LT(index, return_count_);
buffer_[index] = val;
rcursor_ = std::max(rcursor_, index + 1);
}
void AddParam(T val) {
DCHECK_LT(pcursor_, parameter_count_);
buffer_[return_count_ + pcursor_++] = val;
}
void AddParamAt(size_t index, T val) {
DCHECK_LT(index, parameter_count_);
buffer_[return_count_ + index] = val;
pcursor_ = std::max(pcursor_, index + 1);
}
SigT* Get() const {
DCHECK_EQ(rcursor_, return_count_);
DCHECK_EQ(pcursor_, parameter_count_);
DCHECK_NOT_NULL(sig_);
return sig_;
}
protected:
size_t rcursor_;
size_t pcursor_;
SigT* sig_;
T* buffer_;
};
// Describes the inputs and outputs of a function or call.
template <typename T>
class Signature : public ZoneObject {
public:
constexpr Signature(size_t return_count, size_t parameter_count,
const T* reps)
: return_count_(return_count),
parameter_count_(parameter_count),
reps_(reps) {
DCHECK_EQ(kReturnCountOffset, offsetof(Signature, return_count_));
DCHECK_EQ(kParameterCountOffset, offsetof(Signature, parameter_count_));
DCHECK_EQ(kRepsOffset, offsetof(Signature, reps_));
static_assert(std::is_standard_layout_v<Signature<T>>);
}
size_t return_count() const { return return_count_; }
size_t parameter_count() const { return parameter_count_; }
T GetParam(size_t index) const {
// If heap memory is corrupted, we may get confused about the number of
// parameters during compilation. These SBXCHECKs defend against that.
SBXCHECK_LT(index, parameter_count_);
return reps_[return_count_ + index];
}
T GetReturn(size_t index = 0) const {
SBXCHECK_LT(index, return_count_);
return reps_[index];
}
bool contains(T element) const {
return std::find(all().cbegin(), all().cend(), element) != all().cend();
}
// Iteration support.
base::Vector<const T> parameters() const {
return {reps_ + return_count_, parameter_count_};
}
base::Vector<const T> returns() const { return {reps_, return_count_}; }
base::Vector<const T> all() const {
return {reps_, return_count_ + parameter_count_};
}
bool operator==(const Signature& other) const {
if (this == &other) return true;
if (parameter_count() != other.parameter_count()) return false;
if (return_count() != other.return_count()) return false;
return std::equal(all().begin(), all().end(), other.all().begin());
}
bool operator!=(const Signature& other) const { return !(*this == other); }
// For incrementally building signatures.
using Builder = SignatureBuilder<Signature<T>, T>;
static Signature<T>* Build(Zone* zone, std::initializer_list<T> returns,
std::initializer_list<T> params) {
Builder builder(zone, returns.size(), params.size());
for (T ret : returns) builder.AddReturn(ret);
for (T param : params) builder.AddParam(param);
return builder.Get();
}
static constexpr size_t kReturnCountOffset = 0;
static constexpr size_t kParameterCountOffset =
kReturnCountOffset + kSizetSize;
static constexpr size_t kRepsOffset = kParameterCountOffset + kSizetSize;
protected:
size_t return_count_;
size_t parameter_count_;
const T* reps_;
};
using MachineSignature = Signature<MachineType>;
template <typename T>
size_t hash_value(const Signature<T>& sig) {
// Hash over all contained representations, plus the parameter count to
// differentiate signatures with the same representation array but different
// parameter/return count.
return base::Hasher{}.Add(sig.parameter_count()).AddRange(sig.all()).hash();
}
template <typename T, size_t kNumReturns = 0, size_t kNumParams = 0>
class FixedSizeSignature : public Signature<T> {
public:
// Add return types to this signature (only allowed if there are none yet).
template <typename... ReturnTypes>
auto Returns(ReturnTypes... return_types) const {
static_assert(kNumReturns == 0, "Please specify all return types at once");
return FixedSizeSignature<T, sizeof...(ReturnTypes), kNumParams>{
std::initializer_list<T>{return_types...}.begin(), reps_};
}
// Add parameters to this signature (only allowed if there are none yet).
template <typename... ParamTypes>
auto Params(ParamTypes... param_types) const {
static_assert(kNumParams == 0, "Please specify all parameters at once");
return FixedSizeSignature<T, kNumReturns, sizeof...(ParamTypes)>{
reps_, std::initializer_list<T>{param_types...}.begin()};
}
private:
// Other template instantiations can call the private constructor.
template <typename T2, size_t kNumReturns2, size_t kNumParams2>
friend class FixedSizeSignature;
FixedSizeSignature(const T* returns, const T* params)
: Signature<T>(kNumReturns, kNumParams, reps_) {
std::copy(returns, returns + kNumReturns, reps_);
std::copy(params, params + kNumParams, reps_ + kNumReturns);
}
T reps_[kNumReturns + kNumParams];
};
// Specialization for zero-sized signatures.
template <typename T>
class FixedSizeSignature<T, 0, 0> : public Signature<T> {
public:
constexpr FixedSizeSignature() : Signature<T>(0, 0, nullptr) {}
// Add return types.
template <typename... ReturnTypes>
static auto Returns(ReturnTypes... return_types) {
return FixedSizeSignature<T, sizeof...(ReturnTypes), 0>{
std::initializer_list<T>{return_types...}.begin(), nullptr};
}
// Add parameters.
template <typename... ParamTypes>
static auto Params(ParamTypes... param_types) {
return FixedSizeSignature<T, 0, sizeof...(ParamTypes)>{
nullptr, std::initializer_list<T>{param_types...}.begin()};
}
};
} // namespace internal
} // namespace v8
#endif // V8_CODEGEN_SIGNATURE_H_