Add semantic::Intrinsic
semantic::Intrinsic derives from semantic::CallTarget, which can be obtained from the Target() accessor on the CallExpression. Flesh out semantic::Parameter to contain a `Usage` - extra metadata for the parameter. The information in `Intrinsic` is enough to remove the `semantic::IntrinsicCall` and `semantic::TextureIntrinsicCall` types. Change-Id: Ida9c193674ad8605d8f12f6a1d27f38c7d008434 Reviewed-on: https://dawn-review.googlesource.com/c/tint/+/40503 Reviewed-by: dan sinclair <dsinclair@chromium.org> Commit-Queue: Ben Clayton <bclayton@google.com>
This commit is contained in:
parent
bab3197fda
commit
316f9f6b6d
|
@ -26,96 +26,17 @@ namespace semantic {
|
||||||
class Call : public Castable<Call, Expression> {
|
class Call : public Castable<Call, Expression> {
|
||||||
public:
|
public:
|
||||||
/// Constructor
|
/// Constructor
|
||||||
/// @param return_type the return type of the call
|
/// @param target the call target
|
||||||
explicit Call(type::Type* return_type);
|
explicit Call(const CallTarget* target);
|
||||||
|
|
||||||
/// Destructor
|
/// Destructor
|
||||||
~Call() override;
|
~Call() override;
|
||||||
};
|
|
||||||
|
|
||||||
/// IntrinsicCall holds semantic information for ast::CallExpression nodes that
|
/// @return the target of the call
|
||||||
/// call intrinsic functions.
|
const CallTarget* Target() const { return target_; }
|
||||||
class IntrinsicCall : public Castable<IntrinsicCall, Call> {
|
|
||||||
public:
|
|
||||||
/// Constructor
|
|
||||||
/// @param return_type the return type of the call
|
|
||||||
/// @param intrinsic the call target intrinsic
|
|
||||||
IntrinsicCall(type::Type* return_type, IntrinsicType intrinsic);
|
|
||||||
|
|
||||||
/// Destructor
|
|
||||||
~IntrinsicCall() override;
|
|
||||||
|
|
||||||
/// @returns the target intrinsic for the call
|
|
||||||
IntrinsicType intrinsic() const { return intrinsic_; }
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
IntrinsicType const intrinsic_;
|
CallTarget const* const target_;
|
||||||
};
|
|
||||||
|
|
||||||
/// TextureIntrinsicCall holds semantic information for ast::CallExpression
|
|
||||||
/// nodes that call intrinsic texture functions.
|
|
||||||
class TextureIntrinsicCall
|
|
||||||
: public Castable<TextureIntrinsicCall, IntrinsicCall> {
|
|
||||||
public:
|
|
||||||
/// Parameters describes the parameters for the texture function.
|
|
||||||
struct Parameters {
|
|
||||||
/// kNotUsed is the constant that indicates the given parameter is not part
|
|
||||||
/// of the texture function signature.
|
|
||||||
static constexpr const size_t kNotUsed = ~static_cast<size_t>(0u);
|
|
||||||
/// Index holds each of the possible parameter indices. If a parameter index
|
|
||||||
/// is equal to `kNotUsed` then this parameter is not used by the function.
|
|
||||||
struct Index {
|
|
||||||
/// Constructor
|
|
||||||
Index();
|
|
||||||
/// Copy constructor
|
|
||||||
Index(const Index&);
|
|
||||||
/// `array_index` parameter index.
|
|
||||||
size_t array_index = kNotUsed;
|
|
||||||
/// `bias` parameter index.
|
|
||||||
size_t bias = kNotUsed;
|
|
||||||
/// `coords` parameter index.
|
|
||||||
size_t coords = kNotUsed;
|
|
||||||
/// `depth_ref` parameter index.
|
|
||||||
size_t depth_ref = kNotUsed;
|
|
||||||
/// `ddx` parameter index.
|
|
||||||
size_t ddx = kNotUsed;
|
|
||||||
/// `ddy` parameter index.
|
|
||||||
size_t ddy = kNotUsed;
|
|
||||||
/// `level` parameter index.
|
|
||||||
size_t level = kNotUsed;
|
|
||||||
/// `offset` parameter index.
|
|
||||||
size_t offset = kNotUsed;
|
|
||||||
/// `sampler` parameter index.
|
|
||||||
size_t sampler = kNotUsed;
|
|
||||||
/// `sample_index` parameter index.
|
|
||||||
size_t sample_index = kNotUsed;
|
|
||||||
/// `texture` parameter index.
|
|
||||||
size_t texture = kNotUsed;
|
|
||||||
/// `value` parameter index.
|
|
||||||
size_t value = kNotUsed;
|
|
||||||
};
|
|
||||||
/// The indices of all possible parameters.
|
|
||||||
Index idx;
|
|
||||||
/// Total number of parameters.
|
|
||||||
size_t count = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
/// Constructor
|
|
||||||
/// @param return_type the return type of the call
|
|
||||||
/// @param intrinsic the call target intrinsic
|
|
||||||
/// @param params the overload parameter info
|
|
||||||
TextureIntrinsicCall(type::Type* return_type,
|
|
||||||
IntrinsicType intrinsic,
|
|
||||||
const Parameters& params);
|
|
||||||
|
|
||||||
/// Destructor
|
|
||||||
~TextureIntrinsicCall() override;
|
|
||||||
|
|
||||||
/// @return the texture call's parameters
|
|
||||||
const Parameters& Params() const { return params_; }
|
|
||||||
|
|
||||||
private:
|
|
||||||
const Parameters params_;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace semantic
|
} // namespace semantic
|
||||||
|
|
|
@ -32,18 +32,52 @@ namespace semantic {
|
||||||
|
|
||||||
/// Parameter describes a single parameter of a call target
|
/// Parameter describes a single parameter of a call target
|
||||||
struct Parameter {
|
struct Parameter {
|
||||||
|
/// Usage is extra metadata for identifying a parameter based on its overload
|
||||||
|
/// position
|
||||||
|
enum class Usage {
|
||||||
|
kNone,
|
||||||
|
kArrayIndex,
|
||||||
|
kBias,
|
||||||
|
kCoords,
|
||||||
|
kDepthRef,
|
||||||
|
kDdx,
|
||||||
|
kDdy,
|
||||||
|
kLevel,
|
||||||
|
kOffset,
|
||||||
|
kSampler,
|
||||||
|
kSampleIndex,
|
||||||
|
kTexture,
|
||||||
|
kValue,
|
||||||
|
};
|
||||||
|
|
||||||
/// Parameter type
|
/// Parameter type
|
||||||
type::Type* type;
|
type::Type* const type;
|
||||||
|
/// Parameter usage
|
||||||
|
Usage const usage = Usage::kNone;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/// @returns a string representation of the given parameter usage.
|
||||||
|
const char* str(Parameter::Usage usage);
|
||||||
|
|
||||||
|
/// Parameters is a list of Parameter
|
||||||
using Parameters = std::vector<Parameter>;
|
using Parameters = std::vector<Parameter>;
|
||||||
|
|
||||||
|
/// @param parameters the list of parameters
|
||||||
|
/// @param usage the parameter usage to find
|
||||||
|
/// @returns the index of the parameter with the given usage, or -1 if no
|
||||||
|
/// parameter with the given usage exists.
|
||||||
|
int IndexOf(const Parameters& parameters, Parameter::Usage usage);
|
||||||
|
|
||||||
/// CallTarget is the base for callable functions
|
/// CallTarget is the base for callable functions
|
||||||
class CallTarget : public Castable<CallTarget, Node> {
|
class CallTarget : public Castable<CallTarget, Node> {
|
||||||
public:
|
public:
|
||||||
/// Constructor
|
/// Constructor
|
||||||
|
/// @param return_type the return type of the call target
|
||||||
/// @param parameters the parameters for the call target
|
/// @param parameters the parameters for the call target
|
||||||
explicit CallTarget(const semantic::Parameters& parameters);
|
CallTarget(type::Type* return_type, const semantic::Parameters& parameters);
|
||||||
|
|
||||||
|
/// @return the return type of the call target
|
||||||
|
type::Type* ReturnType() const { return return_type_; }
|
||||||
|
|
||||||
/// Destructor
|
/// Destructor
|
||||||
~CallTarget() override;
|
~CallTarget() override;
|
||||||
|
@ -52,7 +86,8 @@ class CallTarget : public Castable<CallTarget, Node> {
|
||||||
const Parameters& Parameters() const { return parameters_; }
|
const Parameters& Parameters() const { return parameters_; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
semantic::Parameters parameters_;
|
type::Type* const return_type_;
|
||||||
|
semantic::Parameters const parameters_;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace semantic
|
} // namespace semantic
|
||||||
|
|
|
@ -17,6 +17,8 @@
|
||||||
|
|
||||||
#include <ostream>
|
#include <ostream>
|
||||||
|
|
||||||
|
#include "src/semantic/call_target.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace semantic {
|
namespace semantic {
|
||||||
|
|
||||||
|
@ -102,39 +104,37 @@ enum class IntrinsicType {
|
||||||
kTrunc
|
kTrunc
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Emits the name of the intrinsic function. The spelling,
|
/// @returns the name of the intrinsic function type. The spelling, including
|
||||||
/// including case, matches the name in the WGSL spec.
|
/// case, matches the name in the WGSL spec.
|
||||||
std::ostream& operator<<(std::ostream& out, IntrinsicType i);
|
const char* str(IntrinsicType i);
|
||||||
|
|
||||||
namespace intrinsic {
|
|
||||||
|
|
||||||
/// Determines if the given `i` is a coarse derivative
|
/// Determines if the given `i` is a coarse derivative
|
||||||
/// @param i the intrinsic
|
/// @param i the intrinsic type
|
||||||
/// @returns true if the given derivative is coarse.
|
/// @returns true if the given derivative is coarse.
|
||||||
bool IsCoarseDerivative(IntrinsicType i);
|
bool IsCoarseDerivativeIntrinsic(IntrinsicType i);
|
||||||
|
|
||||||
/// Determines if the given `i` is a fine derivative
|
/// Determines if the given `i` is a fine derivative
|
||||||
/// @param i the intrinsic
|
/// @param i the intrinsic type
|
||||||
/// @returns true if the given derivative is fine.
|
/// @returns true if the given derivative is fine.
|
||||||
bool IsFineDerivative(IntrinsicType i);
|
bool IsFineDerivativeIntrinsic(IntrinsicType i);
|
||||||
|
|
||||||
/// Determine if the given `i` is a derivative intrinsic
|
/// Determine if the given `i` is a derivative intrinsic
|
||||||
/// @param i the intrinsic
|
/// @param i the intrinsic type
|
||||||
/// @returns true if the given `i` is a derivative intrinsic
|
/// @returns true if the given `i` is a derivative intrinsic
|
||||||
bool IsDerivative(IntrinsicType i);
|
bool IsDerivativeIntrinsic(IntrinsicType i);
|
||||||
|
|
||||||
/// Determines if the given `i` is a float classification intrinsic
|
/// Determines if the given `i` is a float classification intrinsic
|
||||||
/// @param i the intrinsic
|
/// @param i the intrinsic type
|
||||||
/// @returns true if the given `i` is a float intrinsic
|
/// @returns true if the given `i` is a float intrinsic
|
||||||
bool IsFloatClassificationIntrinsic(IntrinsicType i);
|
bool IsFloatClassificationIntrinsic(IntrinsicType i);
|
||||||
|
|
||||||
/// Determines if the given `i` is a texture operation intrinsic
|
/// Determines if the given `i` is a texture operation intrinsic
|
||||||
/// @param i the intrinsic
|
/// @param i the intrinsic type
|
||||||
/// @returns true if the given `i` is a texture operation intrinsic
|
/// @returns true if the given `i` is a texture operation intrinsic
|
||||||
bool IsTextureIntrinsic(IntrinsicType i);
|
bool IsTextureIntrinsic(IntrinsicType i);
|
||||||
|
|
||||||
/// Determines if the given `i` is a image query intrinsic
|
/// Determines if the given `i` is a image query intrinsic
|
||||||
/// @param i the intrinsic
|
/// @param i the intrinsic type
|
||||||
/// @returns true if the given `i` is a image query intrinsic
|
/// @returns true if the given `i` is a image query intrinsic
|
||||||
bool IsImageQueryIntrinsic(IntrinsicType i);
|
bool IsImageQueryIntrinsic(IntrinsicType i);
|
||||||
|
|
||||||
|
@ -143,11 +143,56 @@ bool IsImageQueryIntrinsic(IntrinsicType i);
|
||||||
/// @returns true if the given `i` is a data packing intrinsic
|
/// @returns true if the given `i` is a data packing intrinsic
|
||||||
bool IsDataPackingIntrinsic(IntrinsicType i);
|
bool IsDataPackingIntrinsic(IntrinsicType i);
|
||||||
|
|
||||||
/// @returns the name of the intrinsic function. The spelling, including case,
|
/// Intrinsic holds the semantic information for an intrinsic function.
|
||||||
/// matches the name in the WGSL spec.
|
class Intrinsic : public Castable<Intrinsic, CallTarget> {
|
||||||
const char* str(IntrinsicType i);
|
public:
|
||||||
|
/// Constructor
|
||||||
|
/// @param type the intrinsic type
|
||||||
|
/// @param return_type the return type for the intrinsic call
|
||||||
|
/// @param parameters the parameters for the intrinsic overload
|
||||||
|
Intrinsic(IntrinsicType type,
|
||||||
|
type::Type* return_type,
|
||||||
|
const semantic::Parameters& parameters);
|
||||||
|
|
||||||
|
/// Destructor
|
||||||
|
~Intrinsic() override;
|
||||||
|
|
||||||
|
/// @return the type of the intrinsic
|
||||||
|
IntrinsicType Type() const { return type_; }
|
||||||
|
|
||||||
|
/// @returns the name of the intrinsic function type. The spelling, including
|
||||||
|
/// case, matches the name in the WGSL spec.
|
||||||
|
const char* str() const;
|
||||||
|
|
||||||
|
/// @returns true if intrinsic is a coarse derivative intrinsic
|
||||||
|
bool IsCoarseDerivative() const;
|
||||||
|
|
||||||
|
/// @returns true if intrinsic is a fine a derivative intrinsic
|
||||||
|
bool IsFineDerivative() const;
|
||||||
|
|
||||||
|
/// @returns true if intrinsic is a derivative intrinsic
|
||||||
|
bool IsDerivative() const;
|
||||||
|
|
||||||
|
/// @returns true if intrinsic is a float intrinsic
|
||||||
|
bool IsFloatClassification() const;
|
||||||
|
|
||||||
|
/// @returns true if intrinsic is a texture operation intrinsic
|
||||||
|
bool IsTexture() const;
|
||||||
|
|
||||||
|
/// @returns true if intrinsic is a image query intrinsic
|
||||||
|
bool IsImageQuery() const;
|
||||||
|
|
||||||
|
/// @returns true if intrinsic is a data packing intrinsic
|
||||||
|
bool IsDataPacking() const;
|
||||||
|
|
||||||
|
private:
|
||||||
|
IntrinsicType const type_;
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Emits the name of the intrinsic function type. The spelling, including case,
|
||||||
|
/// matches the name in the WGSL spec.
|
||||||
|
std::ostream& operator<<(std::ostream& out, IntrinsicType i);
|
||||||
|
|
||||||
} // namespace intrinsic
|
|
||||||
} // namespace semantic
|
} // namespace semantic
|
||||||
} // namespace tint
|
} // namespace tint
|
||||||
|
|
||||||
|
|
|
@ -15,30 +15,14 @@
|
||||||
#include "src/semantic/call.h"
|
#include "src/semantic/call.h"
|
||||||
|
|
||||||
TINT_INSTANTIATE_CLASS_ID(tint::semantic::Call);
|
TINT_INSTANTIATE_CLASS_ID(tint::semantic::Call);
|
||||||
TINT_INSTANTIATE_CLASS_ID(tint::semantic::IntrinsicCall);
|
|
||||||
TINT_INSTANTIATE_CLASS_ID(tint::semantic::TextureIntrinsicCall);
|
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace semantic {
|
namespace semantic {
|
||||||
|
|
||||||
Call::Call(type::Type* return_type) : Base(return_type) {}
|
Call::Call(const CallTarget* target)
|
||||||
|
: Base(target->ReturnType()), target_(target) {}
|
||||||
|
|
||||||
Call::~Call() = default;
|
Call::~Call() = default;
|
||||||
|
|
||||||
IntrinsicCall::IntrinsicCall(type::Type* return_type, IntrinsicType intrinsic)
|
|
||||||
: Base(return_type), intrinsic_(intrinsic) {}
|
|
||||||
|
|
||||||
IntrinsicCall::~IntrinsicCall() = default;
|
|
||||||
|
|
||||||
TextureIntrinsicCall::TextureIntrinsicCall(type::Type* return_type,
|
|
||||||
IntrinsicType intrinsic,
|
|
||||||
const Parameters& params)
|
|
||||||
: Base(return_type, intrinsic), params_(params) {}
|
|
||||||
|
|
||||||
TextureIntrinsicCall::~TextureIntrinsicCall() = default;
|
|
||||||
|
|
||||||
TextureIntrinsicCall::Parameters::Index::Index() = default;
|
|
||||||
TextureIntrinsicCall::Parameters::Index::Index(const Index&) = default;
|
|
||||||
|
|
||||||
} // namespace semantic
|
} // namespace semantic
|
||||||
} // namespace tint
|
} // namespace tint
|
||||||
|
|
|
@ -21,10 +21,50 @@ TINT_INSTANTIATE_CLASS_ID(tint::semantic::CallTarget);
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace semantic {
|
namespace semantic {
|
||||||
|
|
||||||
CallTarget::CallTarget(const semantic::Parameters& parameters)
|
CallTarget::CallTarget(type::Type* return_type,
|
||||||
: parameters_(parameters) {}
|
const semantic::Parameters& parameters)
|
||||||
|
: return_type_(return_type), parameters_(parameters) {}
|
||||||
|
|
||||||
CallTarget::~CallTarget() = default;
|
CallTarget::~CallTarget() = default;
|
||||||
|
|
||||||
|
int IndexOf(const Parameters& parameters, Parameter::Usage usage) {
|
||||||
|
for (size_t i = 0; i < parameters.size(); i++) {
|
||||||
|
if (parameters[i].usage == usage) {
|
||||||
|
return static_cast<int>(i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* str(Parameter::Usage usage) {
|
||||||
|
switch (usage) {
|
||||||
|
case Parameter::Usage::kArrayIndex:
|
||||||
|
return "array_index";
|
||||||
|
case Parameter::Usage::kBias:
|
||||||
|
return "bias";
|
||||||
|
case Parameter::Usage::kCoords:
|
||||||
|
return "coords";
|
||||||
|
case Parameter::Usage::kDepthRef:
|
||||||
|
return "depth_ref";
|
||||||
|
case Parameter::Usage::kDdx:
|
||||||
|
return "ddx";
|
||||||
|
case Parameter::Usage::kDdy:
|
||||||
|
return "ddy";
|
||||||
|
case Parameter::Usage::kLevel:
|
||||||
|
return "level";
|
||||||
|
case Parameter::Usage::kOffset:
|
||||||
|
return "offset";
|
||||||
|
case Parameter::Usage::kSampler:
|
||||||
|
return "sampler";
|
||||||
|
case Parameter::Usage::kSampleIndex:
|
||||||
|
return "sample_index";
|
||||||
|
case Parameter::Usage::kTexture:
|
||||||
|
return "texture";
|
||||||
|
case Parameter::Usage::kValue:
|
||||||
|
return "value";
|
||||||
|
default:
|
||||||
|
return "<unknown>";
|
||||||
|
}
|
||||||
|
}
|
||||||
} // namespace semantic
|
} // namespace semantic
|
||||||
} // namespace tint
|
} // namespace tint
|
||||||
|
|
|
@ -37,7 +37,7 @@ Parameters GetParameters(ast::Function* ast) {
|
||||||
semantic::Parameters parameters;
|
semantic::Parameters parameters;
|
||||||
parameters.reserve(ast->params().size());
|
parameters.reserve(ast->params().size());
|
||||||
for (auto* param : ast->params()) {
|
for (auto* param : ast->params()) {
|
||||||
parameters.emplace_back(Parameter{param->type()});
|
parameters.emplace_back(Parameter{param->type(), Parameter::Usage::kNone});
|
||||||
}
|
}
|
||||||
return parameters;
|
return parameters;
|
||||||
}
|
}
|
||||||
|
@ -48,7 +48,7 @@ Function::Function(ast::Function* ast,
|
||||||
std::vector<const Variable*> referenced_module_vars,
|
std::vector<const Variable*> referenced_module_vars,
|
||||||
std::vector<const Variable*> local_referenced_module_vars,
|
std::vector<const Variable*> local_referenced_module_vars,
|
||||||
std::vector<Symbol> ancestor_entry_points)
|
std::vector<Symbol> ancestor_entry_points)
|
||||||
: Base(GetParameters(ast)),
|
: Base(ast->return_type(), GetParameters(ast)),
|
||||||
referenced_module_vars_(std::move(referenced_module_vars)),
|
referenced_module_vars_(std::move(referenced_module_vars)),
|
||||||
local_referenced_module_vars_(std::move(local_referenced_module_vars)),
|
local_referenced_module_vars_(std::move(local_referenced_module_vars)),
|
||||||
ancestor_entry_points_(std::move(ancestor_entry_points)) {}
|
ancestor_entry_points_(std::move(ancestor_entry_points)) {}
|
||||||
|
|
|
@ -14,15 +14,19 @@
|
||||||
|
|
||||||
#include "src/semantic/intrinsic.h"
|
#include "src/semantic/intrinsic.h"
|
||||||
|
|
||||||
|
TINT_INSTANTIATE_CLASS_ID(tint::semantic::Intrinsic);
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
namespace semantic {
|
namespace semantic {
|
||||||
|
|
||||||
std::ostream& operator<<(std::ostream& out, IntrinsicType i) {
|
std::ostream& operator<<(std::ostream& out, IntrinsicType i) {
|
||||||
out << intrinsic::str(i);
|
out << str(i);
|
||||||
return out;
|
return out;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace intrinsic {
|
const char* Intrinsic::str() const {
|
||||||
|
return semantic::str(type_);
|
||||||
|
}
|
||||||
|
|
||||||
const char* str(IntrinsicType i) {
|
const char* str(IntrinsicType i) {
|
||||||
/// The emitted name matches the spelling in the WGSL spec.
|
/// The emitted name matches the spelling in the WGSL spec.
|
||||||
|
@ -188,20 +192,20 @@ const char* str(IntrinsicType i) {
|
||||||
return "<unknown>";
|
return "<unknown>";
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsCoarseDerivative(IntrinsicType i) {
|
bool IsCoarseDerivativeIntrinsic(IntrinsicType i) {
|
||||||
return i == IntrinsicType::kDpdxCoarse || i == IntrinsicType::kDpdyCoarse ||
|
return i == IntrinsicType::kDpdxCoarse || i == IntrinsicType::kDpdyCoarse ||
|
||||||
i == IntrinsicType::kFwidthCoarse;
|
i == IntrinsicType::kFwidthCoarse;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsFineDerivative(IntrinsicType i) {
|
bool IsFineDerivativeIntrinsic(IntrinsicType i) {
|
||||||
return i == IntrinsicType::kDpdxFine || i == IntrinsicType::kDpdyFine ||
|
return i == IntrinsicType::kDpdxFine || i == IntrinsicType::kDpdyFine ||
|
||||||
i == IntrinsicType::kFwidthFine;
|
i == IntrinsicType::kFwidthFine;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsDerivative(IntrinsicType i) {
|
bool IsDerivativeIntrinsic(IntrinsicType i) {
|
||||||
return i == IntrinsicType::kDpdx || i == IntrinsicType::kDpdy ||
|
return i == IntrinsicType::kDpdx || i == IntrinsicType::kDpdy ||
|
||||||
i == IntrinsicType::kFwidth || IsCoarseDerivative(i) ||
|
i == IntrinsicType::kFwidth || IsCoarseDerivativeIntrinsic(i) ||
|
||||||
IsFineDerivative(i);
|
IsFineDerivativeIntrinsic(i);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsFloatClassificationIntrinsic(IntrinsicType i) {
|
bool IsFloatClassificationIntrinsic(IntrinsicType i) {
|
||||||
|
@ -220,7 +224,7 @@ bool IsTextureIntrinsic(IntrinsicType i) {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IsImageQueryIntrinsic(IntrinsicType i) {
|
bool IsImageQueryIntrinsic(IntrinsicType i) {
|
||||||
return i == semantic::IntrinsicType::kTextureDimensions ||
|
return i == IntrinsicType::kTextureDimensions ||
|
||||||
i == IntrinsicType::kTextureNumLayers ||
|
i == IntrinsicType::kTextureNumLayers ||
|
||||||
i == IntrinsicType::kTextureNumLevels ||
|
i == IntrinsicType::kTextureNumLevels ||
|
||||||
i == IntrinsicType::kTextureNumSamples;
|
i == IntrinsicType::kTextureNumSamples;
|
||||||
|
@ -234,6 +238,40 @@ bool IsDataPackingIntrinsic(IntrinsicType i) {
|
||||||
i == IntrinsicType::kPack2x16Float;
|
i == IntrinsicType::kPack2x16Float;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace intrinsic
|
Intrinsic::Intrinsic(IntrinsicType type,
|
||||||
|
type::Type* return_type,
|
||||||
|
const semantic::Parameters& parameters)
|
||||||
|
: Base(return_type, parameters), type_(type) {}
|
||||||
|
|
||||||
|
Intrinsic::~Intrinsic() = default;
|
||||||
|
|
||||||
|
bool Intrinsic::IsCoarseDerivative() const {
|
||||||
|
return IsCoarseDerivativeIntrinsic(type_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Intrinsic::IsFineDerivative() const {
|
||||||
|
return IsFineDerivativeIntrinsic(type_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Intrinsic::IsDerivative() const {
|
||||||
|
return IsDerivativeIntrinsic(type_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Intrinsic::IsFloatClassification() const {
|
||||||
|
return IsFloatClassificationIntrinsic(type_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Intrinsic::IsTexture() const {
|
||||||
|
return IsTextureIntrinsic(type_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Intrinsic::IsImageQuery() const {
|
||||||
|
return IsImageQueryIntrinsic(type_);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Intrinsic::IsDataPacking() const {
|
||||||
|
return IsDataPackingIntrinsic(type_);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace semantic
|
} // namespace semantic
|
||||||
} // namespace tint
|
} // namespace tint
|
||||||
|
|
|
@ -419,9 +419,9 @@ bool TypeDeterminer::DetermineCall(ast::CallExpression* call) {
|
||||||
|
|
||||||
auto name = builder_->Symbols().NameFor(ident->symbol());
|
auto name = builder_->Symbols().NameFor(ident->symbol());
|
||||||
|
|
||||||
auto intrinsic = MatchIntrinsic(name);
|
auto intrinsic_type = MatchIntrinsicType(name);
|
||||||
if (intrinsic != IntrinsicType::kNone) {
|
if (intrinsic_type != IntrinsicType::kNone) {
|
||||||
if (!DetermineIntrinsicCall(call, intrinsic)) {
|
if (!DetermineIntrinsicCall(call, intrinsic_type)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
@ -450,9 +450,7 @@ bool TypeDeterminer::DetermineCall(ast::CallExpression* call) {
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* function = iter->second;
|
auto* function = iter->second;
|
||||||
auto* return_ty = function->declaration->return_type();
|
function_calls_.emplace(call, function);
|
||||||
auto* sem = builder_->create<semantic::Call>(return_ty);
|
|
||||||
builder_->Sem().Add(call, sem);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
@ -548,14 +546,26 @@ constexpr const uint32_t kIntrinsicDataCount =
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
bool TypeDeterminer::DetermineIntrinsicCall(ast::CallExpression* call,
|
bool TypeDeterminer::DetermineIntrinsicCall(ast::CallExpression* call,
|
||||||
IntrinsicType intrinsic) {
|
IntrinsicType intrinsic_type) {
|
||||||
auto create_sem = [&](type::Type* result) {
|
using Parameter = semantic::Parameter;
|
||||||
auto* sem = builder_->create<semantic::IntrinsicCall>(result, intrinsic);
|
using Parameters = semantic::Parameters;
|
||||||
builder_->Sem().Add(call, sem);
|
using Usage = Parameter::Usage;
|
||||||
|
|
||||||
|
std::vector<type::Type*> arg_tys;
|
||||||
|
arg_tys.reserve(call->params().size());
|
||||||
|
for (auto* expr : call->params()) {
|
||||||
|
arg_tys.emplace_back(TypeOf(expr));
|
||||||
|
}
|
||||||
|
|
||||||
|
auto create_sem = [&](type::Type* return_type) {
|
||||||
|
semantic::Parameters params; // TODO(bclayton): Populate this
|
||||||
|
auto* intrinsic = builder_->create<semantic::Intrinsic>(
|
||||||
|
intrinsic_type, return_type, params);
|
||||||
|
builder_->Sem().Add(call, builder_->create<semantic::Call>(intrinsic));
|
||||||
};
|
};
|
||||||
|
|
||||||
std::string name = semantic::intrinsic::str(intrinsic);
|
std::string name = semantic::str(intrinsic_type);
|
||||||
if (semantic::intrinsic::IsFloatClassificationIntrinsic(intrinsic)) {
|
if (semantic::IsFloatClassificationIntrinsic(intrinsic_type)) {
|
||||||
if (call->params().size() != 1) {
|
if (call->params().size() != 1) {
|
||||||
set_error(call->source(), "incorrect number of parameters for " + name);
|
set_error(call->source(), "incorrect number of parameters for " + name);
|
||||||
return false;
|
return false;
|
||||||
|
@ -571,131 +581,135 @@ bool TypeDeterminer::DetermineIntrinsicCall(ast::CallExpression* call,
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (semantic::intrinsic::IsTextureIntrinsic(intrinsic)) {
|
if (semantic::IsTextureIntrinsic(intrinsic_type)) {
|
||||||
semantic::TextureIntrinsicCall::Parameters param;
|
Parameters params;
|
||||||
|
|
||||||
auto* texture_param = call->params()[0];
|
auto& ty = builder_->ty;
|
||||||
if (!TypeOf(texture_param)->UnwrapAll()->Is<type::Texture>()) {
|
|
||||||
|
auto* texture = arg_tys[0]->UnwrapAll()->As<type::Texture>();
|
||||||
|
if (!texture) {
|
||||||
set_error(call->source(), "invalid first argument for " + name);
|
set_error(call->source(), "invalid first argument for " + name);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
type::Texture* texture =
|
|
||||||
TypeOf(texture_param)->UnwrapAll()->As<type::Texture>();
|
|
||||||
|
|
||||||
bool is_array = type::IsTextureArray(texture->dim());
|
bool is_array = type::IsTextureArray(texture->dim());
|
||||||
bool is_multisampled = texture->Is<type::MultisampledTexture>();
|
bool is_multisampled = texture->Is<type::MultisampledTexture>();
|
||||||
switch (intrinsic) {
|
switch (intrinsic_type) {
|
||||||
case IntrinsicType::kTextureDimensions:
|
case IntrinsicType::kTextureDimensions:
|
||||||
param.idx.texture = param.count++;
|
params.emplace_back(Parameter{texture, Usage::kTexture});
|
||||||
if (call->params().size() > param.count) {
|
if (arg_tys.size() > params.size()) {
|
||||||
param.idx.level = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kLevel});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case IntrinsicType::kTextureNumLayers:
|
case IntrinsicType::kTextureNumLayers:
|
||||||
case IntrinsicType::kTextureNumLevels:
|
case IntrinsicType::kTextureNumLevels:
|
||||||
case IntrinsicType::kTextureNumSamples:
|
case IntrinsicType::kTextureNumSamples:
|
||||||
param.idx.texture = param.count++;
|
params.emplace_back(Parameter{texture, Usage::kTexture});
|
||||||
break;
|
break;
|
||||||
case IntrinsicType::kTextureLoad:
|
case IntrinsicType::kTextureLoad:
|
||||||
param.idx.texture = param.count++;
|
params.emplace_back(Parameter{texture, Usage::kTexture});
|
||||||
param.idx.coords = param.count++;
|
params.emplace_back(Parameter{arg_tys[1], Usage::kCoords});
|
||||||
if (is_array) {
|
if (is_array) {
|
||||||
param.idx.array_index = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kArrayIndex});
|
||||||
}
|
}
|
||||||
if (call->params().size() > param.count) {
|
if (arg_tys.size() > params.size()) {
|
||||||
if (is_multisampled) {
|
if (is_multisampled) {
|
||||||
param.idx.sample_index = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kSampleIndex});
|
||||||
} else {
|
} else {
|
||||||
param.idx.level = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kLevel});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case IntrinsicType::kTextureSample:
|
case IntrinsicType::kTextureSample:
|
||||||
param.idx.texture = param.count++;
|
params.emplace_back(Parameter{texture, Usage::kTexture});
|
||||||
param.idx.sampler = param.count++;
|
params.emplace_back(Parameter{arg_tys[1], Usage::kSampler});
|
||||||
param.idx.coords = param.count++;
|
params.emplace_back(Parameter{arg_tys[2], Usage::kCoords});
|
||||||
if (is_array) {
|
if (is_array) {
|
||||||
param.idx.array_index = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kArrayIndex});
|
||||||
}
|
}
|
||||||
if (call->params().size() > param.count) {
|
if (arg_tys.size() > params.size()) {
|
||||||
param.idx.offset = param.count++;
|
params.emplace_back(
|
||||||
|
Parameter{arg_tys[params.size()], Usage::kOffset});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case IntrinsicType::kTextureSampleBias:
|
case IntrinsicType::kTextureSampleBias:
|
||||||
param.idx.texture = param.count++;
|
params.emplace_back(Parameter{texture, Usage::kTexture});
|
||||||
param.idx.sampler = param.count++;
|
params.emplace_back(Parameter{arg_tys[1], Usage::kSampler});
|
||||||
param.idx.coords = param.count++;
|
params.emplace_back(Parameter{arg_tys[2], Usage::kCoords});
|
||||||
if (is_array) {
|
if (is_array) {
|
||||||
param.idx.array_index = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kArrayIndex});
|
||||||
}
|
}
|
||||||
param.idx.bias = param.count++;
|
params.emplace_back(Parameter{ty.f32(), Usage::kBias});
|
||||||
if (call->params().size() > param.count) {
|
if (arg_tys.size() > params.size()) {
|
||||||
param.idx.offset = param.count++;
|
params.emplace_back(
|
||||||
|
Parameter{arg_tys[params.size()], Usage::kOffset});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case IntrinsicType::kTextureSampleLevel:
|
case IntrinsicType::kTextureSampleLevel:
|
||||||
param.idx.texture = param.count++;
|
params.emplace_back(Parameter{texture, Usage::kTexture});
|
||||||
param.idx.sampler = param.count++;
|
params.emplace_back(Parameter{arg_tys[1], Usage::kSampler});
|
||||||
param.idx.coords = param.count++;
|
params.emplace_back(Parameter{arg_tys[2], Usage::kCoords});
|
||||||
if (is_array) {
|
if (is_array) {
|
||||||
param.idx.array_index = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kArrayIndex});
|
||||||
}
|
}
|
||||||
param.idx.level = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kLevel});
|
||||||
if (call->params().size() > param.count) {
|
if (arg_tys.size() > params.size()) {
|
||||||
param.idx.offset = param.count++;
|
params.emplace_back(
|
||||||
|
Parameter{arg_tys[params.size()], Usage::kOffset});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case IntrinsicType::kTextureSampleCompare:
|
case IntrinsicType::kTextureSampleCompare:
|
||||||
param.idx.texture = param.count++;
|
params.emplace_back(Parameter{texture, Usage::kTexture});
|
||||||
param.idx.sampler = param.count++;
|
params.emplace_back(Parameter{arg_tys[1], Usage::kSampler});
|
||||||
param.idx.coords = param.count++;
|
params.emplace_back(Parameter{arg_tys[2], Usage::kCoords});
|
||||||
if (is_array) {
|
if (is_array) {
|
||||||
param.idx.array_index = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kArrayIndex});
|
||||||
}
|
}
|
||||||
param.idx.depth_ref = param.count++;
|
params.emplace_back(Parameter{ty.f32(), Usage::kDepthRef});
|
||||||
if (call->params().size() > param.count) {
|
if (arg_tys.size() > params.size()) {
|
||||||
param.idx.offset = param.count++;
|
params.emplace_back(
|
||||||
|
Parameter{arg_tys[params.size()], Usage::kOffset});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case IntrinsicType::kTextureSampleGrad:
|
case IntrinsicType::kTextureSampleGrad:
|
||||||
param.idx.texture = param.count++;
|
params.emplace_back(Parameter{texture, Usage::kTexture});
|
||||||
param.idx.sampler = param.count++;
|
params.emplace_back(Parameter{arg_tys[1], Usage::kSampler});
|
||||||
param.idx.coords = param.count++;
|
params.emplace_back(Parameter{arg_tys[2], Usage::kCoords});
|
||||||
if (is_array) {
|
if (is_array) {
|
||||||
param.idx.array_index = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kArrayIndex});
|
||||||
}
|
}
|
||||||
param.idx.ddx = param.count++;
|
params.emplace_back(Parameter{arg_tys[params.size()], Usage::kDdx});
|
||||||
param.idx.ddy = param.count++;
|
params.emplace_back(Parameter{arg_tys[params.size()], Usage::kDdy});
|
||||||
if (call->params().size() > param.count) {
|
if (arg_tys.size() > params.size()) {
|
||||||
param.idx.offset = param.count++;
|
params.emplace_back(
|
||||||
|
Parameter{arg_tys[params.size()], Usage::kOffset});
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case IntrinsicType::kTextureStore:
|
case IntrinsicType::kTextureStore:
|
||||||
param.idx.texture = param.count++;
|
params.emplace_back(Parameter{texture, Usage::kTexture});
|
||||||
param.idx.coords = param.count++;
|
params.emplace_back(Parameter{arg_tys[1], Usage::kCoords});
|
||||||
if (is_array) {
|
if (is_array) {
|
||||||
param.idx.array_index = param.count++;
|
params.emplace_back(Parameter{ty.i32(), Usage::kArrayIndex});
|
||||||
}
|
}
|
||||||
param.idx.value = param.count++;
|
params.emplace_back(Parameter{arg_tys[params.size()], Usage::kValue});
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
set_error(call->source(),
|
set_error(call->source(),
|
||||||
"Internal compiler error: Unreachable intrinsic " +
|
"Internal compiler error: Unreachable intrinsic " + name);
|
||||||
std::to_string(static_cast<int>(intrinsic)));
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (call->params().size() != param.count) {
|
if (arg_tys.size() != params.size()) {
|
||||||
set_error(call->source(),
|
set_error(call->source(), "incorrect number of arguments for " + name +
|
||||||
"incorrect number of parameters for " + name + ", got " +
|
", got " + std::to_string(arg_tys.size()) +
|
||||||
std::to_string(call->params().size()) + " and expected " +
|
" and expected " +
|
||||||
std::to_string(param.count));
|
std::to_string(params.size()));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set the function return type
|
// Set the function return type
|
||||||
type::Type* return_type = nullptr;
|
type::Type* return_type = nullptr;
|
||||||
switch (intrinsic) {
|
switch (intrinsic_type) {
|
||||||
case IntrinsicType::kTextureDimensions: {
|
case IntrinsicType::kTextureDimensions: {
|
||||||
auto* i32 = builder_->create<type::I32>();
|
auto* i32 = builder_->create<type::I32>();
|
||||||
switch (texture->dim()) {
|
switch (texture->dim()) {
|
||||||
|
@ -748,16 +762,15 @@ bool TypeDeterminer::DetermineIntrinsicCall(ast::CallExpression* call,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* sem = builder_->create<semantic::TextureIntrinsicCall>(
|
auto* intrinsic = builder_->create<semantic::Intrinsic>(
|
||||||
return_type, intrinsic, param);
|
intrinsic_type, return_type, params);
|
||||||
builder_->Sem().Add(call, sem);
|
builder_->Sem().Add(call, builder_->create<semantic::Call>(intrinsic));
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
const IntrinsicData* data = nullptr;
|
const IntrinsicData* data = nullptr;
|
||||||
for (uint32_t i = 0; i < kIntrinsicDataCount; ++i) {
|
for (uint32_t i = 0; i < kIntrinsicDataCount; ++i) {
|
||||||
if (intrinsic == kIntrinsicData[i].intrinsic) {
|
if (intrinsic_type == kIntrinsicData[i].intrinsic) {
|
||||||
data = &kIntrinsicData[i];
|
data = &kIntrinsicData[i];
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -847,7 +860,7 @@ bool TypeDeterminer::DetermineIdentifier(ast::IdentifierExpression* expr) {
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string name = builder_->Symbols().NameFor(symbol);
|
std::string name = builder_->Symbols().NameFor(symbol);
|
||||||
if (MatchIntrinsic(name) != IntrinsicType::kNone) {
|
if (MatchIntrinsicType(name) != IntrinsicType::kNone) {
|
||||||
// Identifier is to an intrinsic function, which has no type (currently).
|
// Identifier is to an intrinsic function, which has no type (currently).
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -857,7 +870,7 @@ bool TypeDeterminer::DetermineIdentifier(ast::IdentifierExpression* expr) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
IntrinsicType TypeDeterminer::MatchIntrinsic(const std::string& name) {
|
IntrinsicType TypeDeterminer::MatchIntrinsicType(const std::string& name) {
|
||||||
if (name == "abs") {
|
if (name == "abs") {
|
||||||
return IntrinsicType::kAbs;
|
return IntrinsicType::kAbs;
|
||||||
} else if (name == "acos") {
|
} else if (name == "acos") {
|
||||||
|
@ -1197,14 +1210,23 @@ void TypeDeterminer::CreateSemanticNodes() const {
|
||||||
return out;
|
return out;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
std::unordered_map<FunctionInfo*, semantic::Function*> func_info_to_sem_func;
|
||||||
for (auto it : function_to_info_) {
|
for (auto it : function_to_info_) {
|
||||||
auto* func = it.first;
|
auto* func = it.first;
|
||||||
auto* info = it.second;
|
auto* info = it.second;
|
||||||
sem.Add(func,
|
auto* sem_func = builder_->create<semantic::Function>(
|
||||||
builder_->create<semantic::Function>(
|
info->declaration, remap_vars(info->referenced_module_vars),
|
||||||
info->declaration, remap_vars(info->referenced_module_vars),
|
remap_vars(info->local_referenced_module_vars),
|
||||||
remap_vars(info->local_referenced_module_vars),
|
info->ancestor_entry_points);
|
||||||
info->ancestor_entry_points));
|
func_info_to_sem_func.emplace(info, sem_func);
|
||||||
|
sem.Add(func, sem_func);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (auto it : function_calls_) {
|
||||||
|
auto* call = it.first;
|
||||||
|
auto* func_info = it.second;
|
||||||
|
auto* sem_func = func_info_to_sem_func.at(func_info);
|
||||||
|
builder_->Sem().Add(call, builder_->create<semantic::Call>(sem_func));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,6 +24,7 @@
|
||||||
#include "src/diagnostic/diagnostic.h"
|
#include "src/diagnostic/diagnostic.h"
|
||||||
#include "src/program_builder.h"
|
#include "src/program_builder.h"
|
||||||
#include "src/scope_stack.h"
|
#include "src/scope_stack.h"
|
||||||
|
#include "src/semantic/intrinsic.h"
|
||||||
#include "src/type/storage_texture_type.h"
|
#include "src/type/storage_texture_type.h"
|
||||||
|
|
||||||
namespace tint {
|
namespace tint {
|
||||||
|
@ -66,10 +67,10 @@ class TypeDeterminer {
|
||||||
/// @returns true if the type determiner was successful
|
/// @returns true if the type determiner was successful
|
||||||
bool Determine();
|
bool Determine();
|
||||||
|
|
||||||
/// @param name the function name to try and match as an intrinsic.
|
/// @param name the function name to try and match as an intrinsic type.
|
||||||
/// @return the semantic::IntrinsicType for the given name. If `name` does not
|
/// @return the semantic::IntrinsicType for the given name. If `name` does not
|
||||||
/// match an intrinsic, returns semantic::IntrinsicType::kNone
|
/// match an intrinsic, returns semantic::Intrinsic::kNone
|
||||||
static semantic::IntrinsicType MatchIntrinsic(const std::string& name);
|
static semantic::IntrinsicType MatchIntrinsicType(const std::string& name);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
template <typename T>
|
template <typename T>
|
||||||
|
@ -177,7 +178,7 @@ class TypeDeterminer {
|
||||||
bool DetermineConstructor(ast::ConstructorExpression* expr);
|
bool DetermineConstructor(ast::ConstructorExpression* expr);
|
||||||
bool DetermineIdentifier(ast::IdentifierExpression* expr);
|
bool DetermineIdentifier(ast::IdentifierExpression* expr);
|
||||||
bool DetermineIntrinsicCall(ast::CallExpression* call,
|
bool DetermineIntrinsicCall(ast::CallExpression* call,
|
||||||
semantic::IntrinsicType intrinsic);
|
semantic::IntrinsicType intrinsic_type);
|
||||||
bool DetermineMemberAccessor(ast::MemberAccessorExpression* expr);
|
bool DetermineMemberAccessor(ast::MemberAccessorExpression* expr);
|
||||||
bool DetermineUnaryOp(ast::UnaryOpExpression* expr);
|
bool DetermineUnaryOp(ast::UnaryOpExpression* expr);
|
||||||
|
|
||||||
|
@ -195,12 +196,13 @@ class TypeDeterminer {
|
||||||
/// @param type the resolved type
|
/// @param type the resolved type
|
||||||
void SetType(ast::Expression* expr, type::Type* type) const;
|
void SetType(ast::Expression* expr, type::Type* type) const;
|
||||||
|
|
||||||
ProgramBuilder* builder_;
|
ProgramBuilder* const builder_;
|
||||||
std::string error_;
|
std::string error_;
|
||||||
ScopeStack<VariableInfo*> variable_stack_;
|
ScopeStack<VariableInfo*> variable_stack_;
|
||||||
std::unordered_map<Symbol, FunctionInfo*> symbol_to_function_;
|
std::unordered_map<Symbol, FunctionInfo*> symbol_to_function_;
|
||||||
std::unordered_map<ast::Function*, FunctionInfo*> function_to_info_;
|
std::unordered_map<ast::Function*, FunctionInfo*> function_to_info_;
|
||||||
std::unordered_map<ast::Variable*, VariableInfo*> variable_to_info_;
|
std::unordered_map<ast::Variable*, VariableInfo*> variable_to_info_;
|
||||||
|
std::unordered_map<ast::CallExpression*, FunctionInfo*> function_calls_;
|
||||||
FunctionInfo* current_function_ = nullptr;
|
FunctionInfo* current_function_ = nullptr;
|
||||||
BlockAllocator<VariableInfo> variable_infos_;
|
BlockAllocator<VariableInfo> variable_infos_;
|
||||||
BlockAllocator<FunctionInfo> function_infos_;
|
BlockAllocator<FunctionInfo> function_infos_;
|
||||||
|
|
|
@ -1637,7 +1637,7 @@ using IntrinsicDataTest = TypeDeterminerTestWithParam<IntrinsicData>;
|
||||||
TEST_P(IntrinsicDataTest, Lookup) {
|
TEST_P(IntrinsicDataTest, Lookup) {
|
||||||
auto param = GetParam();
|
auto param = GetParam();
|
||||||
|
|
||||||
EXPECT_EQ(TypeDeterminer::MatchIntrinsic(param.name), param.intrinsic);
|
EXPECT_EQ(TypeDeterminer::MatchIntrinsicType(param.name), param.intrinsic);
|
||||||
}
|
}
|
||||||
INSTANTIATE_TEST_SUITE_P(
|
INSTANTIATE_TEST_SUITE_P(
|
||||||
TypeDeterminerTest,
|
TypeDeterminerTest,
|
||||||
|
@ -1717,7 +1717,7 @@ INSTANTIATE_TEST_SUITE_P(
|
||||||
IntrinsicData{"trunc", IntrinsicType::kTrunc}));
|
IntrinsicData{"trunc", IntrinsicType::kTrunc}));
|
||||||
|
|
||||||
TEST_F(TypeDeterminerTest, MatchIntrinsicNoMatch) {
|
TEST_F(TypeDeterminerTest, MatchIntrinsicNoMatch) {
|
||||||
EXPECT_EQ(TypeDeterminer::MatchIntrinsic("not_intrinsic"),
|
EXPECT_EQ(TypeDeterminer::MatchIntrinsicType("not_intrinsic"),
|
||||||
IntrinsicType::kNone);
|
IntrinsicType::kNone);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2466,42 +2466,15 @@ INSTANTIATE_TEST_SUITE_P(
|
||||||
testing::ValuesIn(ast::intrinsic::test::TextureOverloadCase::ValidCases()));
|
testing::ValuesIn(ast::intrinsic::test::TextureOverloadCase::ValidCases()));
|
||||||
|
|
||||||
std::string to_str(const std::string& function,
|
std::string to_str(const std::string& function,
|
||||||
const semantic::TextureIntrinsicCall::Parameters& params) {
|
const semantic::Parameters& params) {
|
||||||
struct Parameter {
|
|
||||||
size_t idx;
|
|
||||||
std::string name;
|
|
||||||
};
|
|
||||||
std::vector<Parameter> list;
|
|
||||||
auto maybe_add_param = [&list](size_t idx, const char* name) {
|
|
||||||
if (idx !=
|
|
||||||
semantic::TextureIntrinsicCall::Parameters::Parameters::kNotUsed) {
|
|
||||||
list.emplace_back(Parameter{idx, name});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
maybe_add_param(params.idx.array_index, "array_index");
|
|
||||||
maybe_add_param(params.idx.bias, "bias");
|
|
||||||
maybe_add_param(params.idx.coords, "coords");
|
|
||||||
maybe_add_param(params.idx.depth_ref, "depth_ref");
|
|
||||||
maybe_add_param(params.idx.ddx, "ddx");
|
|
||||||
maybe_add_param(params.idx.ddy, "ddy");
|
|
||||||
maybe_add_param(params.idx.level, "level");
|
|
||||||
maybe_add_param(params.idx.offset, "offset");
|
|
||||||
maybe_add_param(params.idx.sampler, "sampler");
|
|
||||||
maybe_add_param(params.idx.sample_index, "sample_index");
|
|
||||||
maybe_add_param(params.idx.texture, "texture");
|
|
||||||
maybe_add_param(params.idx.value, "value");
|
|
||||||
std::sort(
|
|
||||||
list.begin(), list.end(),
|
|
||||||
[](const Parameter& a, const Parameter& b) { return a.idx < b.idx; });
|
|
||||||
|
|
||||||
std::stringstream out;
|
std::stringstream out;
|
||||||
out << function << "(";
|
out << function << "(";
|
||||||
bool first = true;
|
bool first = true;
|
||||||
for (auto& param : list) {
|
for (auto& param : params) {
|
||||||
if (!first) {
|
if (!first) {
|
||||||
out << ", ";
|
out << ", ";
|
||||||
}
|
}
|
||||||
out << param.name;
|
out << semantic::str(param.usage);
|
||||||
first = false;
|
first = false;
|
||||||
}
|
}
|
||||||
out << ")";
|
out << ")";
|
||||||
|
@ -2833,12 +2806,12 @@ TEST_P(TypeDeterminerTextureIntrinsicTest, Call) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* sem = Sem().Get(call);
|
auto* call_sem = Sem().Get(call);
|
||||||
ASSERT_NE(sem, nullptr);
|
ASSERT_NE(call_sem, nullptr);
|
||||||
auto* intrinsic = sem->As<semantic::TextureIntrinsicCall>();
|
auto* target = call_sem->Target();
|
||||||
ASSERT_NE(intrinsic, nullptr);
|
ASSERT_NE(target, nullptr);
|
||||||
|
|
||||||
auto got = ::tint::to_str(param.function, intrinsic->Params());
|
auto got = ::tint::to_str(param.function, target->Parameters());
|
||||||
auto* expected = expected_texture_overload(param.overload);
|
auto* expected = expected_texture_overload(param.overload);
|
||||||
EXPECT_EQ(got, expected);
|
EXPECT_EQ(got, expected);
|
||||||
}
|
}
|
||||||
|
|
|
@ -644,24 +644,23 @@ bool ValidatorImpl::ValidateCallExpr(const ast::CallExpression* expr) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* call_sem = program_->Sem().Get(expr);
|
auto* call = program_->Sem().Get(expr);
|
||||||
if (call_sem == nullptr) {
|
if (call == nullptr) {
|
||||||
add_error(expr->source(), "CallExpression is missing semantic information");
|
add_error(expr->source(), "CallExpression is missing semantic information");
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (auto* intrinsic_sem = call_sem->As<semantic::IntrinsicCall>()) {
|
if (auto* intrinsic = call->Target()->As<semantic::Intrinsic>()) {
|
||||||
const IntrinsicData* data = nullptr;
|
const IntrinsicData* data = nullptr;
|
||||||
for (uint32_t i = 0; i < kIntrinsicDataCount; ++i) {
|
for (uint32_t i = 0; i < kIntrinsicDataCount; ++i) {
|
||||||
if (intrinsic_sem->intrinsic() == kIntrinsicData[i].intrinsic) {
|
if (intrinsic->Type() == kIntrinsicData[i].intrinsic) {
|
||||||
data = &kIntrinsicData[i];
|
data = &kIntrinsicData[i];
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (data != nullptr) {
|
if (data != nullptr) {
|
||||||
std::string builtin =
|
std::string builtin = intrinsic->str();
|
||||||
semantic::intrinsic::str(intrinsic_sem->intrinsic());
|
|
||||||
if (expr->params().size() != data->param_count) {
|
if (expr->params().size() != data->param_count) {
|
||||||
add_error(expr->source(),
|
add_error(expr->source(),
|
||||||
"incorrect number of parameters for " + builtin +
|
"incorrect number of parameters for " + builtin +
|
||||||
|
@ -832,7 +831,7 @@ bool ValidatorImpl::ValidateCallExpr(const ast::CallExpression* expr) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (semantic::intrinsic::IsDataPackingIntrinsic(data->intrinsic)) {
|
if (semantic::IsDataPackingIntrinsic(data->intrinsic)) {
|
||||||
if (!program_->TypeOf(expr)->Is<type::U32>()) {
|
if (!program_->TypeOf(expr)->Is<type::U32>()) {
|
||||||
add_error(expr->source(),
|
add_error(expr->source(),
|
||||||
"incorrect type for " + builtin +
|
"incorrect type for " + builtin +
|
||||||
|
|
|
@ -542,22 +542,22 @@ bool GeneratorImpl::EmitCall(std::ostream& pre,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* call_sem = builder_.Sem().Get(expr);
|
auto* call = builder_.Sem().Get(expr);
|
||||||
if (auto* sem = call_sem->As<semantic::TextureIntrinsicCall>()) {
|
if (auto* intrinsic = call->Target()->As<semantic::Intrinsic>()) {
|
||||||
return EmitTextureCall(pre, out, expr, sem);
|
if (intrinsic->IsTexture()) {
|
||||||
}
|
return EmitTextureCall(pre, out, expr, intrinsic);
|
||||||
if (auto* sem = call_sem->As<semantic::IntrinsicCall>()) {
|
}
|
||||||
const auto& params = expr->params();
|
const auto& params = expr->params();
|
||||||
if (sem->intrinsic() == semantic::IntrinsicType::kSelect) {
|
if (intrinsic->Type() == semantic::IntrinsicType::kSelect) {
|
||||||
error_ = "select not supported in HLSL backend yet";
|
error_ = "select not supported in HLSL backend yet";
|
||||||
return false;
|
return false;
|
||||||
} else if (sem->intrinsic() == semantic::IntrinsicType::kIsNormal) {
|
} else if (intrinsic->Type() == semantic::IntrinsicType::kIsNormal) {
|
||||||
error_ = "is_normal not supported in HLSL backend yet";
|
error_ = "is_normal not supported in HLSL backend yet";
|
||||||
return false;
|
return false;
|
||||||
} else if (semantic::intrinsic::IsDataPackingIntrinsic(sem->intrinsic())) {
|
} else if (intrinsic->IsDataPacking()) {
|
||||||
return EmitDataPackingCall(pre, out, expr);
|
return EmitDataPackingCall(pre, out, expr, intrinsic);
|
||||||
}
|
}
|
||||||
auto name = generate_builtin_name(sem);
|
auto name = generate_builtin_name(intrinsic);
|
||||||
if (name.empty()) {
|
if (name.empty()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -638,9 +638,8 @@ bool GeneratorImpl::EmitCall(std::ostream& pre,
|
||||||
|
|
||||||
bool GeneratorImpl::EmitDataPackingCall(std::ostream& pre,
|
bool GeneratorImpl::EmitDataPackingCall(std::ostream& pre,
|
||||||
std::ostream& out,
|
std::ostream& out,
|
||||||
ast::CallExpression* expr) {
|
ast::CallExpression* expr,
|
||||||
auto* ident = builder_.Sem().Get(expr)->As<semantic::IntrinsicCall>();
|
const semantic::Intrinsic* intrinsic) {
|
||||||
|
|
||||||
auto* param = expr->params()[0];
|
auto* param = expr->params()[0];
|
||||||
auto tmp_name = generate_name(kTempNamePrefix);
|
auto tmp_name = generate_name(kTempNamePrefix);
|
||||||
std::ostringstream expr_out;
|
std::ostringstream expr_out;
|
||||||
|
@ -650,17 +649,17 @@ bool GeneratorImpl::EmitDataPackingCall(std::ostream& pre,
|
||||||
uint32_t dims = 2;
|
uint32_t dims = 2;
|
||||||
bool is_signed = false;
|
bool is_signed = false;
|
||||||
uint32_t scale = 65535;
|
uint32_t scale = 65535;
|
||||||
if (ident->intrinsic() == semantic::IntrinsicType::kPack4x8Snorm ||
|
if (intrinsic->Type() == semantic::IntrinsicType::kPack4x8Snorm ||
|
||||||
ident->intrinsic() == semantic::IntrinsicType::kPack4x8Unorm) {
|
intrinsic->Type() == semantic::IntrinsicType::kPack4x8Unorm) {
|
||||||
dims = 4;
|
dims = 4;
|
||||||
scale = 255;
|
scale = 255;
|
||||||
}
|
}
|
||||||
if (ident->intrinsic() == semantic::IntrinsicType::kPack4x8Snorm ||
|
if (intrinsic->Type() == semantic::IntrinsicType::kPack4x8Snorm ||
|
||||||
ident->intrinsic() == semantic::IntrinsicType::kPack2x16Snorm) {
|
intrinsic->Type() == semantic::IntrinsicType::kPack2x16Snorm) {
|
||||||
is_signed = true;
|
is_signed = true;
|
||||||
scale = (scale - 1) / 2;
|
scale = (scale - 1) / 2;
|
||||||
}
|
}
|
||||||
switch (ident->intrinsic()) {
|
switch (intrinsic->Type()) {
|
||||||
case semantic::IntrinsicType::kPack4x8Snorm:
|
case semantic::IntrinsicType::kPack4x8Snorm:
|
||||||
case semantic::IntrinsicType::kPack4x8Unorm:
|
case semantic::IntrinsicType::kPack4x8Unorm:
|
||||||
case semantic::IntrinsicType::kPack2x16Snorm:
|
case semantic::IntrinsicType::kPack2x16Snorm:
|
||||||
|
@ -698,17 +697,24 @@ bool GeneratorImpl::EmitDataPackingCall(std::ostream& pre,
|
||||||
bool GeneratorImpl::EmitTextureCall(std::ostream& pre,
|
bool GeneratorImpl::EmitTextureCall(std::ostream& pre,
|
||||||
std::ostream& out,
|
std::ostream& out,
|
||||||
ast::CallExpression* expr,
|
ast::CallExpression* expr,
|
||||||
const semantic::TextureIntrinsicCall* sem) {
|
const semantic::Intrinsic* intrinsic) {
|
||||||
auto* ident = expr->func()->As<ast::IdentifierExpression>();
|
using Usage = semantic::Parameter::Usage;
|
||||||
|
|
||||||
auto params = expr->params();
|
auto parameters = intrinsic->Parameters();
|
||||||
auto& pidx = sem->Params().idx;
|
auto arguments = expr->params();
|
||||||
auto const kNotUsed = semantic::TextureIntrinsicCall::Parameters::kNotUsed;
|
|
||||||
|
// Returns the argument with the given usage
|
||||||
|
auto arg = [&](Usage usage) {
|
||||||
|
int idx = semantic::IndexOf(parameters, usage);
|
||||||
|
return (idx >= 0) ? arguments[idx] : nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
auto* texture = arg(Usage::kTexture);
|
||||||
|
assert(texture);
|
||||||
|
|
||||||
auto* texture = params[pidx.texture];
|
|
||||||
auto* texture_type = TypeOf(texture)->UnwrapAll()->As<type::Texture>();
|
auto* texture_type = TypeOf(texture)->UnwrapAll()->As<type::Texture>();
|
||||||
|
|
||||||
switch (sem->intrinsic()) {
|
switch (intrinsic->Type()) {
|
||||||
case semantic::IntrinsicType::kTextureDimensions:
|
case semantic::IntrinsicType::kTextureDimensions:
|
||||||
case semantic::IntrinsicType::kTextureNumLayers:
|
case semantic::IntrinsicType::kTextureNumLayers:
|
||||||
case semantic::IntrinsicType::kTextureNumLevels:
|
case semantic::IntrinsicType::kTextureNumLevels:
|
||||||
|
@ -718,7 +724,7 @@ bool GeneratorImpl::EmitTextureCall(std::ostream& pre,
|
||||||
const char* swizzle = "";
|
const char* swizzle = "";
|
||||||
bool add_mip_level_in = false;
|
bool add_mip_level_in = false;
|
||||||
|
|
||||||
switch (sem->intrinsic()) {
|
switch (intrinsic->Type()) {
|
||||||
case semantic::IntrinsicType::kTextureDimensions:
|
case semantic::IntrinsicType::kTextureDimensions:
|
||||||
switch (texture_type->dim()) {
|
switch (texture_type->dim()) {
|
||||||
case type::TextureDimension::kNone:
|
case type::TextureDimension::kNone:
|
||||||
|
@ -823,8 +829,11 @@ bool GeneratorImpl::EmitTextureCall(std::ostream& pre,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
pre << ".GetDimensions(";
|
pre << ".GetDimensions(";
|
||||||
if (pidx.level != kNotUsed) {
|
if (auto* level = arg(Usage::kLevel)) {
|
||||||
pre << pidx.level << ", ";
|
if (!EmitExpression(pre, pre, level)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
pre << ", ";
|
||||||
} else if (add_mip_level_in) {
|
} else if (add_mip_level_in) {
|
||||||
pre << "0, ";
|
pre << "0, ";
|
||||||
}
|
}
|
||||||
|
@ -859,7 +868,7 @@ bool GeneratorImpl::EmitTextureCall(std::ostream& pre,
|
||||||
|
|
||||||
bool pack_mip_in_coords = false;
|
bool pack_mip_in_coords = false;
|
||||||
|
|
||||||
switch (sem->intrinsic()) {
|
switch (intrinsic->Type()) {
|
||||||
case semantic::IntrinsicType::kTextureSample:
|
case semantic::IntrinsicType::kTextureSample:
|
||||||
out << ".Sample(";
|
out << ".Sample(";
|
||||||
break;
|
break;
|
||||||
|
@ -886,17 +895,18 @@ bool GeneratorImpl::EmitTextureCall(std::ostream& pre,
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
error_ = "Internal compiler error: Unhandled texture intrinsic '" +
|
error_ = "Internal compiler error: Unhandled texture intrinsic '" +
|
||||||
builder_.Symbols().NameFor(ident->symbol()) + "'";
|
std::string(intrinsic->str()) + "'";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pidx.sampler != kNotUsed) {
|
if (auto* sampler = arg(Usage::kSampler)) {
|
||||||
if (!EmitExpression(pre, out, params[pidx.sampler]))
|
if (!EmitExpression(pre, out, sampler))
|
||||||
return false;
|
return false;
|
||||||
out << ", ";
|
out << ", ";
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* param_coords = params[pidx.coords];
|
auto* param_coords = arg(Usage::kCoords);
|
||||||
|
assert(param_coords);
|
||||||
|
|
||||||
auto emit_vector_appended_with_i32_zero = [&](tint::ast::Expression* vector) {
|
auto emit_vector_appended_with_i32_zero = [&](tint::ast::Expression* vector) {
|
||||||
auto* i32 = builder_.create<type::I32>();
|
auto* i32 = builder_.create<type::I32>();
|
||||||
|
@ -906,10 +916,9 @@ bool GeneratorImpl::EmitTextureCall(std::ostream& pre,
|
||||||
return EmitExpression(pre, out, packed);
|
return EmitExpression(pre, out, packed);
|
||||||
};
|
};
|
||||||
|
|
||||||
if (pidx.array_index != kNotUsed) {
|
if (auto* array_index = arg(Usage::kArrayIndex)) {
|
||||||
// Array index needs to be appended to the coordinates.
|
// Array index needs to be appended to the coordinates.
|
||||||
auto* param_array_index = params[pidx.array_index];
|
auto* packed = AppendVector(&builder_, param_coords, array_index);
|
||||||
auto* packed = AppendVector(&builder_, param_coords, param_array_index);
|
|
||||||
if (pack_mip_in_coords) {
|
if (pack_mip_in_coords) {
|
||||||
if (!emit_vector_appended_with_i32_zero(packed)) {
|
if (!emit_vector_appended_with_i32_zero(packed)) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -928,18 +937,18 @@ bool GeneratorImpl::EmitTextureCall(std::ostream& pre,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto idx : {pidx.depth_ref, pidx.bias, pidx.level, pidx.ddx, pidx.ddy,
|
for (auto usage : {Usage::kDepthRef, Usage::kBias, Usage::kLevel, Usage::kDdx,
|
||||||
pidx.sample_index, pidx.offset}) {
|
Usage::kDdy, Usage::kSampleIndex, Usage::kOffset}) {
|
||||||
if (idx != kNotUsed) {
|
if (auto* e = arg(usage)) {
|
||||||
out << ", ";
|
out << ", ";
|
||||||
if (!EmitExpression(pre, out, params[idx]))
|
if (!EmitExpression(pre, out, e))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sem->intrinsic() == semantic::IntrinsicType::kTextureStore) {
|
if (intrinsic->Type() == semantic::IntrinsicType::kTextureStore) {
|
||||||
out << "] = ";
|
out << "] = ";
|
||||||
if (!EmitExpression(pre, out, params[pidx.value]))
|
if (!EmitExpression(pre, out, arg(Usage::kValue)))
|
||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
out << ")";
|
out << ")";
|
||||||
|
@ -949,9 +958,9 @@ bool GeneratorImpl::EmitTextureCall(std::ostream& pre,
|
||||||
} // namespace hlsl
|
} // namespace hlsl
|
||||||
|
|
||||||
std::string GeneratorImpl::generate_builtin_name(
|
std::string GeneratorImpl::generate_builtin_name(
|
||||||
const semantic::IntrinsicCall* call) {
|
const semantic::Intrinsic* intrinsic) {
|
||||||
std::string out;
|
std::string out;
|
||||||
switch (call->intrinsic()) {
|
switch (intrinsic->Type()) {
|
||||||
case semantic::IntrinsicType::kAcos:
|
case semantic::IntrinsicType::kAcos:
|
||||||
case semantic::IntrinsicType::kAny:
|
case semantic::IntrinsicType::kAny:
|
||||||
case semantic::IntrinsicType::kAll:
|
case semantic::IntrinsicType::kAll:
|
||||||
|
@ -990,7 +999,7 @@ std::string GeneratorImpl::generate_builtin_name(
|
||||||
case semantic::IntrinsicType::kMax:
|
case semantic::IntrinsicType::kMax:
|
||||||
case semantic::IntrinsicType::kMin:
|
case semantic::IntrinsicType::kMin:
|
||||||
case semantic::IntrinsicType::kClamp:
|
case semantic::IntrinsicType::kClamp:
|
||||||
out = semantic::intrinsic::str(call->intrinsic());
|
out = intrinsic->str();
|
||||||
break;
|
break;
|
||||||
case semantic::IntrinsicType::kCountOneBits:
|
case semantic::IntrinsicType::kCountOneBits:
|
||||||
out = "countbits";
|
out = "countbits";
|
||||||
|
@ -1043,8 +1052,7 @@ std::string GeneratorImpl::generate_builtin_name(
|
||||||
out = "smoothstep";
|
out = "smoothstep";
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
error_ = "Unknown builtin method: " +
|
error_ = "Unknown builtin method: " + std::string(intrinsic->str());
|
||||||
std::string(semantic::intrinsic::str(call->intrinsic()));
|
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -48,8 +48,8 @@ namespace tint {
|
||||||
|
|
||||||
// Forward declarations
|
// Forward declarations
|
||||||
namespace semantic {
|
namespace semantic {
|
||||||
class TextureIntrinsicCall;
|
class Call;
|
||||||
class IntrinsicCall;
|
class Intrinsic;
|
||||||
} // namespace semantic
|
} // namespace semantic
|
||||||
|
|
||||||
namespace writer {
|
namespace writer {
|
||||||
|
@ -153,20 +153,22 @@ class GeneratorImpl {
|
||||||
/// @param pre the preamble for the expression stream
|
/// @param pre the preamble for the expression stream
|
||||||
/// @param out the output of the expression stream
|
/// @param out the output of the expression stream
|
||||||
/// @param expr the call expression
|
/// @param expr the call expression
|
||||||
/// @param sem the semantic information for the texture intrinsic call
|
/// @param intrinsic the semantic information for the texture intrinsic
|
||||||
/// @returns true if the call expression is emitted
|
/// @returns true if the call expression is emitted
|
||||||
bool EmitTextureCall(std::ostream& pre,
|
bool EmitTextureCall(std::ostream& pre,
|
||||||
std::ostream& out,
|
std::ostream& out,
|
||||||
ast::CallExpression* expr,
|
ast::CallExpression* expr,
|
||||||
const semantic::TextureIntrinsicCall* sem);
|
const semantic::Intrinsic* intrinsic);
|
||||||
/// Handles generating a call to data packing intrinsic
|
/// Handles generating a call to data packing intrinsic
|
||||||
/// @param pre the preamble of the expression stream
|
/// @param pre the preamble of the expression stream
|
||||||
/// @param out the output of the expression stream
|
/// @param out the output of the expression stream
|
||||||
/// @param expr the call expression
|
/// @param expr the call expression
|
||||||
|
/// @param intrinsic the semantic information for the texture intrinsic
|
||||||
/// @returns true if the call expression is emitted
|
/// @returns true if the call expression is emitted
|
||||||
bool EmitDataPackingCall(std::ostream& pre,
|
bool EmitDataPackingCall(std::ostream& pre,
|
||||||
std::ostream& out,
|
std::ostream& out,
|
||||||
ast::CallExpression* expr);
|
ast::CallExpression* expr,
|
||||||
|
const semantic::Intrinsic* intrinsic);
|
||||||
/// Handles a case statement
|
/// Handles a case statement
|
||||||
/// @param out the output stream
|
/// @param out the output stream
|
||||||
/// @param stmt the statement
|
/// @param stmt the statement
|
||||||
|
@ -363,9 +365,9 @@ class GeneratorImpl {
|
||||||
std::string generate_storage_buffer_index_expression(std::ostream& pre,
|
std::string generate_storage_buffer_index_expression(std::ostream& pre,
|
||||||
ast::Expression* expr);
|
ast::Expression* expr);
|
||||||
/// Handles generating a builtin method name
|
/// Handles generating a builtin method name
|
||||||
/// @param call the semantic info for the intrinsic call
|
/// @param intrinsic the semantic info for the intrinsic
|
||||||
/// @returns the name or "" if not valid
|
/// @returns the name or "" if not valid
|
||||||
std::string generate_builtin_name(const semantic::IntrinsicCall* call);
|
std::string generate_builtin_name(const semantic::Intrinsic* intrinsic);
|
||||||
/// Converts a builtin to an attribute name
|
/// Converts a builtin to an attribute name
|
||||||
/// @param builtin the builtin to convert
|
/// @param builtin the builtin to convert
|
||||||
/// @returns the string name of the builtin or blank on error
|
/// @returns the string name of the builtin or blank on error
|
||||||
|
|
|
@ -175,7 +175,9 @@ TEST_P(HlslIntrinsicTest, Emit) {
|
||||||
|
|
||||||
auto* sem = program->Sem().Get(call);
|
auto* sem = program->Sem().Get(call);
|
||||||
ASSERT_NE(sem, nullptr);
|
ASSERT_NE(sem, nullptr);
|
||||||
auto* intrinsic = sem->As<semantic::IntrinsicCall>();
|
auto* target = sem->Target();
|
||||||
|
ASSERT_NE(target, nullptr);
|
||||||
|
auto* intrinsic = target->As<semantic::Intrinsic>();
|
||||||
ASSERT_NE(intrinsic, nullptr);
|
ASSERT_NE(intrinsic, nullptr);
|
||||||
|
|
||||||
EXPECT_EQ(gen.generate_builtin_name(intrinsic), param.hlsl_name);
|
EXPECT_EQ(gen.generate_builtin_name(intrinsic), param.hlsl_name);
|
||||||
|
|
|
@ -444,12 +444,12 @@ bool GeneratorImpl::EmitCall(ast::CallExpression* expr) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* call_sem = program_->Sem().Get(expr);
|
auto* call = program_->Sem().Get(expr);
|
||||||
if (auto* sem = call_sem->As<semantic::TextureIntrinsicCall>()) {
|
if (auto* intrinsic = call->Target()->As<semantic::Intrinsic>()) {
|
||||||
return EmitTextureCall(expr, sem);
|
if (intrinsic->IsTexture()) {
|
||||||
}
|
return EmitTextureCall(expr, intrinsic);
|
||||||
if (auto* sem = call_sem->As<semantic::IntrinsicCall>()) {
|
}
|
||||||
if (sem->intrinsic() == semantic::IntrinsicType::kPack2x16Float) {
|
if (intrinsic->Type() == semantic::IntrinsicType::kPack2x16Float) {
|
||||||
make_indent();
|
make_indent();
|
||||||
out_ << "as_type<uint>(half2(";
|
out_ << "as_type<uint>(half2(";
|
||||||
if (!EmitExpression(expr->params()[0])) {
|
if (!EmitExpression(expr->params()[0])) {
|
||||||
|
@ -458,7 +458,7 @@ bool GeneratorImpl::EmitCall(ast::CallExpression* expr) {
|
||||||
out_ << "))";
|
out_ << "))";
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
auto name = generate_builtin_name(sem);
|
auto name = generate_builtin_name(intrinsic);
|
||||||
if (name.empty()) {
|
if (name.empty()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -568,18 +568,24 @@ bool GeneratorImpl::EmitCall(ast::CallExpression* expr) {
|
||||||
}
|
}
|
||||||
|
|
||||||
bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
const semantic::TextureIntrinsicCall* sem) {
|
const semantic::Intrinsic* intrinsic) {
|
||||||
auto* ident = expr->func()->As<ast::IdentifierExpression>();
|
using Usage = semantic::Parameter::Usage;
|
||||||
|
|
||||||
auto params = expr->params();
|
auto parameters = intrinsic->Parameters();
|
||||||
auto& pidx = sem->Params().idx;
|
auto arguments = expr->params();
|
||||||
auto const kNotUsed = semantic::TextureIntrinsicCall::Parameters::kNotUsed;
|
|
||||||
|
|
||||||
assert(pidx.texture != kNotUsed);
|
// Returns the argument with the given usage
|
||||||
auto* texture_type =
|
auto arg = [&](Usage usage) {
|
||||||
TypeOf(params[pidx.texture])->UnwrapAll()->As<type::Texture>();
|
int idx = semantic::IndexOf(parameters, usage);
|
||||||
|
return (idx >= 0) ? arguments[idx] : nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
switch (sem->intrinsic()) {
|
auto* texture = arg(Usage::kTexture);
|
||||||
|
assert(texture);
|
||||||
|
|
||||||
|
auto* texture_type = TypeOf(texture)->UnwrapAll()->As<type::Texture>();
|
||||||
|
|
||||||
|
switch (intrinsic->Type()) {
|
||||||
case semantic::IntrinsicType::kTextureDimensions: {
|
case semantic::IntrinsicType::kTextureDimensions: {
|
||||||
std::vector<const char*> dims;
|
std::vector<const char*> dims;
|
||||||
switch (texture_type->dim()) {
|
switch (texture_type->dim()) {
|
||||||
|
@ -606,12 +612,14 @@ bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
}
|
}
|
||||||
|
|
||||||
auto get_dim = [&](const char* name) {
|
auto get_dim = [&](const char* name) {
|
||||||
if (!EmitExpression(params[pidx.texture])) {
|
if (!EmitExpression(texture)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
out_ << ".get_" << name << "(";
|
out_ << ".get_" << name << "(";
|
||||||
if (pidx.level != kNotUsed) {
|
if (auto* level = arg(Usage::kLevel)) {
|
||||||
out_ << pidx.level;
|
if (!EmitExpression(level)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
out_ << ")";
|
out_ << ")";
|
||||||
return true;
|
return true;
|
||||||
|
@ -636,7 +644,7 @@ bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
}
|
}
|
||||||
case semantic::IntrinsicType::kTextureNumLayers: {
|
case semantic::IntrinsicType::kTextureNumLayers: {
|
||||||
out_ << "int(";
|
out_ << "int(";
|
||||||
if (!EmitExpression(params[pidx.texture])) {
|
if (!EmitExpression(texture)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
out_ << ".get_array_size())";
|
out_ << ".get_array_size())";
|
||||||
|
@ -644,7 +652,7 @@ bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
}
|
}
|
||||||
case semantic::IntrinsicType::kTextureNumLevels: {
|
case semantic::IntrinsicType::kTextureNumLevels: {
|
||||||
out_ << "int(";
|
out_ << "int(";
|
||||||
if (!EmitExpression(params[pidx.texture])) {
|
if (!EmitExpression(texture)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
out_ << ".get_num_mip_levels())";
|
out_ << ".get_num_mip_levels())";
|
||||||
|
@ -652,7 +660,7 @@ bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
}
|
}
|
||||||
case semantic::IntrinsicType::kTextureNumSamples: {
|
case semantic::IntrinsicType::kTextureNumSamples: {
|
||||||
out_ << "int(";
|
out_ << "int(";
|
||||||
if (!EmitExpression(params[pidx.texture])) {
|
if (!EmitExpression(texture)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
out_ << ".get_num_samples())";
|
out_ << ".get_num_samples())";
|
||||||
|
@ -662,12 +670,12 @@ bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!EmitExpression(params[pidx.texture]))
|
if (!EmitExpression(texture))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
bool lod_param_is_named = true;
|
bool lod_param_is_named = true;
|
||||||
|
|
||||||
switch (sem->intrinsic()) {
|
switch (intrinsic->Type()) {
|
||||||
case semantic::IntrinsicType::kTextureSample:
|
case semantic::IntrinsicType::kTextureSample:
|
||||||
case semantic::IntrinsicType::kTextureSampleBias:
|
case semantic::IntrinsicType::kTextureSampleBias:
|
||||||
case semantic::IntrinsicType::kTextureSampleLevel:
|
case semantic::IntrinsicType::kTextureSampleLevel:
|
||||||
|
@ -686,7 +694,7 @@ bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
error_ = "Internal compiler error: Unhandled texture intrinsic '" +
|
error_ = "Internal compiler error: Unhandled texture intrinsic '" +
|
||||||
program_->Symbols().NameFor(ident->symbol()) + "'";
|
std::string(intrinsic->str()) + "'";
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -698,40 +706,38 @@ bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
first_arg = false;
|
first_arg = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
for (auto idx : {pidx.value, pidx.sampler, pidx.coords, pidx.array_index,
|
for (auto usage :
|
||||||
pidx.depth_ref, pidx.sample_index}) {
|
{Usage::kValue, Usage::kSampler, Usage::kCoords, Usage::kArrayIndex,
|
||||||
if (idx != kNotUsed) {
|
Usage::kDepthRef, Usage::kSampleIndex}) {
|
||||||
|
if (auto* e = arg(usage)) {
|
||||||
maybe_write_comma();
|
maybe_write_comma();
|
||||||
if (!EmitExpression(params[idx]))
|
if (!EmitExpression(e))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pidx.bias != kNotUsed) {
|
if (auto* bias = arg(Usage::kBias)) {
|
||||||
maybe_write_comma();
|
maybe_write_comma();
|
||||||
out_ << "bias(";
|
out_ << "bias(";
|
||||||
if (!EmitExpression(params[pidx.bias])) {
|
if (!EmitExpression(bias)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
out_ << ")";
|
out_ << ")";
|
||||||
}
|
}
|
||||||
if (pidx.level != kNotUsed) {
|
if (auto* level = arg(Usage::kLevel)) {
|
||||||
maybe_write_comma();
|
maybe_write_comma();
|
||||||
if (lod_param_is_named) {
|
if (lod_param_is_named) {
|
||||||
out_ << "level(";
|
out_ << "level(";
|
||||||
}
|
}
|
||||||
if (!EmitExpression(params[pidx.level])) {
|
if (!EmitExpression(level)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (lod_param_is_named) {
|
if (lod_param_is_named) {
|
||||||
out_ << ")";
|
out_ << ")";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (pidx.ddx != kNotUsed) {
|
if (auto* ddx = arg(Usage::kDdx)) {
|
||||||
auto dim = TypeOf(params[pidx.texture])
|
auto dim = texture_type->dim();
|
||||||
->UnwrapPtrIfNeeded()
|
|
||||||
->As<type::Texture>()
|
|
||||||
->dim();
|
|
||||||
switch (dim) {
|
switch (dim) {
|
||||||
case type::TextureDimension::k2d:
|
case type::TextureDimension::k2d:
|
||||||
case type::TextureDimension::k2dArray:
|
case type::TextureDimension::k2dArray:
|
||||||
|
@ -754,19 +760,19 @@ bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!EmitExpression(params[pidx.ddx])) {
|
if (!EmitExpression(ddx)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
out_ << ", ";
|
out_ << ", ";
|
||||||
if (!EmitExpression(params[pidx.ddy])) {
|
if (!EmitExpression(arg(Usage::kDdy))) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
out_ << ")";
|
out_ << ")";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pidx.offset != kNotUsed) {
|
if (auto* offset = arg(Usage::kOffset)) {
|
||||||
maybe_write_comma();
|
maybe_write_comma();
|
||||||
if (!EmitExpression(params[pidx.offset])) {
|
if (!EmitExpression(offset)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -777,9 +783,9 @@ bool GeneratorImpl::EmitTextureCall(ast::CallExpression* expr,
|
||||||
}
|
}
|
||||||
|
|
||||||
std::string GeneratorImpl::generate_builtin_name(
|
std::string GeneratorImpl::generate_builtin_name(
|
||||||
const semantic::IntrinsicCall* call) {
|
const semantic::Intrinsic* intrinsic) {
|
||||||
std::string out = "metal::";
|
std::string out = "metal::";
|
||||||
switch (call->intrinsic()) {
|
switch (intrinsic->Type()) {
|
||||||
case semantic::IntrinsicType::kAcos:
|
case semantic::IntrinsicType::kAcos:
|
||||||
case semantic::IntrinsicType::kAll:
|
case semantic::IntrinsicType::kAll:
|
||||||
case semantic::IntrinsicType::kAny:
|
case semantic::IntrinsicType::kAny:
|
||||||
|
@ -817,10 +823,10 @@ std::string GeneratorImpl::generate_builtin_name(
|
||||||
case semantic::IntrinsicType::kTrunc:
|
case semantic::IntrinsicType::kTrunc:
|
||||||
case semantic::IntrinsicType::kSign:
|
case semantic::IntrinsicType::kSign:
|
||||||
case semantic::IntrinsicType::kClamp:
|
case semantic::IntrinsicType::kClamp:
|
||||||
out += semantic::intrinsic::str(call->intrinsic());
|
out += intrinsic->str();
|
||||||
break;
|
break;
|
||||||
case semantic::IntrinsicType::kAbs:
|
case semantic::IntrinsicType::kAbs:
|
||||||
if (call->Type()->is_float_scalar_or_vector()) {
|
if (intrinsic->ReturnType()->is_float_scalar_or_vector()) {
|
||||||
out += "fabs";
|
out += "fabs";
|
||||||
} else {
|
} else {
|
||||||
out += "abs";
|
out += "abs";
|
||||||
|
@ -857,14 +863,14 @@ std::string GeneratorImpl::generate_builtin_name(
|
||||||
out += "isnormal";
|
out += "isnormal";
|
||||||
break;
|
break;
|
||||||
case semantic::IntrinsicType::kMax:
|
case semantic::IntrinsicType::kMax:
|
||||||
if (call->Type()->is_float_scalar_or_vector()) {
|
if (intrinsic->ReturnType()->is_float_scalar_or_vector()) {
|
||||||
out += "fmax";
|
out += "fmax";
|
||||||
} else {
|
} else {
|
||||||
out += "max";
|
out += "max";
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case semantic::IntrinsicType::kMin:
|
case semantic::IntrinsicType::kMin:
|
||||||
if (call->Type()->is_float_scalar_or_vector()) {
|
if (intrinsic->ReturnType()->is_float_scalar_or_vector()) {
|
||||||
out += "fmin";
|
out += "fmin";
|
||||||
} else {
|
} else {
|
||||||
out += "min";
|
out += "min";
|
||||||
|
@ -895,8 +901,7 @@ std::string GeneratorImpl::generate_builtin_name(
|
||||||
out += "rsqrt";
|
out += "rsqrt";
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
error_ = "Unknown import method: " +
|
error_ = "Unknown import method: " + std::string(intrinsic->str());
|
||||||
std::string(semantic::intrinsic::str(call->intrinsic()));
|
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
return out;
|
return out;
|
||||||
|
|
|
@ -50,8 +50,8 @@ namespace tint {
|
||||||
|
|
||||||
// Forward declarations
|
// Forward declarations
|
||||||
namespace semantic {
|
namespace semantic {
|
||||||
class TextureIntrinsicCall;
|
class Call;
|
||||||
class IntrinsicCall;
|
class Intrinsic;
|
||||||
} // namespace semantic
|
} // namespace semantic
|
||||||
|
|
||||||
namespace writer {
|
namespace writer {
|
||||||
|
@ -121,10 +121,10 @@ class GeneratorImpl : public TextGenerator {
|
||||||
/// Handles generating a call to a texture function (`textureSample`,
|
/// Handles generating a call to a texture function (`textureSample`,
|
||||||
/// `textureSampleGrad`, etc)
|
/// `textureSampleGrad`, etc)
|
||||||
/// @param expr the call expression
|
/// @param expr the call expression
|
||||||
/// @param sem the semantic information for the texture intrinsic call
|
/// @param intrinsic the semantic information for the texture intrinsic
|
||||||
/// @returns true if the call expression is emitted
|
/// @returns true if the call expression is emitted
|
||||||
bool EmitTextureCall(ast::CallExpression* expr,
|
bool EmitTextureCall(ast::CallExpression* expr,
|
||||||
const semantic::TextureIntrinsicCall* sem);
|
const semantic::Intrinsic* intrinsic);
|
||||||
/// Handles a case statement
|
/// Handles a case statement
|
||||||
/// @param stmt the statement
|
/// @param stmt the statement
|
||||||
/// @returns true if the statement was emitted successfully
|
/// @returns true if the statement was emitted successfully
|
||||||
|
@ -258,9 +258,9 @@ class GeneratorImpl : public TextGenerator {
|
||||||
/// @returns the name
|
/// @returns the name
|
||||||
std::string generate_name(const std::string& prefix);
|
std::string generate_name(const std::string& prefix);
|
||||||
/// Handles generating a builtin name
|
/// Handles generating a builtin name
|
||||||
/// @param call the semantic info for the intrinsic call
|
/// @param intrinsic the semantic info for the intrinsic
|
||||||
/// @returns the name or "" if not valid
|
/// @returns the name or "" if not valid
|
||||||
std::string generate_builtin_name(const semantic::IntrinsicCall* call);
|
std::string generate_builtin_name(const semantic::Intrinsic* intrinsic);
|
||||||
|
|
||||||
/// Checks if the global variable is in an input or output struct
|
/// Checks if the global variable is in an input or output struct
|
||||||
/// @param var the variable to check
|
/// @param var the variable to check
|
||||||
|
|
|
@ -60,7 +60,9 @@ TEST_P(MslImportData_SingleParamTest, FloatScalar) {
|
||||||
|
|
||||||
auto* sem = program->Sem().Get(call);
|
auto* sem = program->Sem().Get(call);
|
||||||
ASSERT_NE(sem, nullptr);
|
ASSERT_NE(sem, nullptr);
|
||||||
auto* intrinsic = sem->As<semantic::IntrinsicCall>();
|
auto* target = sem->Target();
|
||||||
|
ASSERT_NE(target, nullptr);
|
||||||
|
auto* intrinsic = target->As<semantic::Intrinsic>();
|
||||||
ASSERT_NE(intrinsic, nullptr);
|
ASSERT_NE(intrinsic, nullptr);
|
||||||
|
|
||||||
ASSERT_EQ(gen.generate_builtin_name(intrinsic),
|
ASSERT_EQ(gen.generate_builtin_name(intrinsic),
|
||||||
|
|
|
@ -182,7 +182,9 @@ TEST_P(MslIntrinsicTest, Emit) {
|
||||||
|
|
||||||
auto* sem = program->Sem().Get(call);
|
auto* sem = program->Sem().Get(call);
|
||||||
ASSERT_NE(sem, nullptr);
|
ASSERT_NE(sem, nullptr);
|
||||||
auto* intrinsic = sem->As<semantic::IntrinsicCall>();
|
auto* target = sem->Target();
|
||||||
|
ASSERT_NE(target, nullptr);
|
||||||
|
auto* intrinsic = target->As<semantic::Intrinsic>();
|
||||||
ASSERT_NE(intrinsic, nullptr);
|
ASSERT_NE(intrinsic, nullptr);
|
||||||
|
|
||||||
EXPECT_EQ(gen.generate_builtin_name(intrinsic), param.msl_name);
|
EXPECT_EQ(gen.generate_builtin_name(intrinsic), param.msl_name);
|
||||||
|
|
|
@ -168,10 +168,10 @@ type::Matrix* GetNestedMatrixType(type::Type* type) {
|
||||||
return type->As<type::Matrix>();
|
return type->As<type::Matrix>();
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t intrinsic_to_glsl_method(type::Type* type, IntrinsicType intrinsic) {
|
uint32_t intrinsic_to_glsl_method(const semantic::Intrinsic* intrinsic) {
|
||||||
switch (intrinsic) {
|
switch (intrinsic->Type()) {
|
||||||
case IntrinsicType::kAbs:
|
case IntrinsicType::kAbs:
|
||||||
if (type->is_float_scalar_or_vector()) {
|
if (intrinsic->ReturnType()->is_float_scalar_or_vector()) {
|
||||||
return GLSLstd450FAbs;
|
return GLSLstd450FAbs;
|
||||||
} else {
|
} else {
|
||||||
return GLSLstd450SAbs;
|
return GLSLstd450SAbs;
|
||||||
|
@ -187,9 +187,9 @@ uint32_t intrinsic_to_glsl_method(type::Type* type, IntrinsicType intrinsic) {
|
||||||
case IntrinsicType::kCeil:
|
case IntrinsicType::kCeil:
|
||||||
return GLSLstd450Ceil;
|
return GLSLstd450Ceil;
|
||||||
case IntrinsicType::kClamp:
|
case IntrinsicType::kClamp:
|
||||||
if (type->is_float_scalar_or_vector()) {
|
if (intrinsic->ReturnType()->is_float_scalar_or_vector()) {
|
||||||
return GLSLstd450NClamp;
|
return GLSLstd450NClamp;
|
||||||
} else if (type->is_unsigned_scalar_or_vector()) {
|
} else if (intrinsic->ReturnType()->is_unsigned_scalar_or_vector()) {
|
||||||
return GLSLstd450UClamp;
|
return GLSLstd450UClamp;
|
||||||
} else {
|
} else {
|
||||||
return GLSLstd450SClamp;
|
return GLSLstd450SClamp;
|
||||||
|
@ -229,17 +229,17 @@ uint32_t intrinsic_to_glsl_method(type::Type* type, IntrinsicType intrinsic) {
|
||||||
case IntrinsicType::kLog2:
|
case IntrinsicType::kLog2:
|
||||||
return GLSLstd450Log2;
|
return GLSLstd450Log2;
|
||||||
case IntrinsicType::kMax:
|
case IntrinsicType::kMax:
|
||||||
if (type->is_float_scalar_or_vector()) {
|
if (intrinsic->ReturnType()->is_float_scalar_or_vector()) {
|
||||||
return GLSLstd450NMax;
|
return GLSLstd450NMax;
|
||||||
} else if (type->is_unsigned_scalar_or_vector()) {
|
} else if (intrinsic->ReturnType()->is_unsigned_scalar_or_vector()) {
|
||||||
return GLSLstd450UMax;
|
return GLSLstd450UMax;
|
||||||
} else {
|
} else {
|
||||||
return GLSLstd450SMax;
|
return GLSLstd450SMax;
|
||||||
}
|
}
|
||||||
case IntrinsicType::kMin:
|
case IntrinsicType::kMin:
|
||||||
if (type->is_float_scalar_or_vector()) {
|
if (intrinsic->ReturnType()->is_float_scalar_or_vector()) {
|
||||||
return GLSLstd450NMin;
|
return GLSLstd450NMin;
|
||||||
} else if (type->is_unsigned_scalar_or_vector()) {
|
} else if (intrinsic->ReturnType()->is_unsigned_scalar_or_vector()) {
|
||||||
return GLSLstd450UMin;
|
return GLSLstd450UMin;
|
||||||
} else {
|
} else {
|
||||||
return GLSLstd450SMin;
|
return GLSLstd450SMin;
|
||||||
|
@ -1826,12 +1826,13 @@ uint32_t Builder::GenerateCallExpression(ast::CallExpression* expr) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* sem = builder_.Sem().Get(expr);
|
auto* call = builder_.Sem().Get(expr);
|
||||||
if (auto* intrinsic = sem->As<semantic::IntrinsicCall>()) {
|
auto* target = call->Target();
|
||||||
return GenerateIntrinsic(ident, expr, intrinsic);
|
if (auto* intrinsic = target->As<semantic::Intrinsic>()) {
|
||||||
|
return GenerateIntrinsic(expr, intrinsic);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto type_id = GenerateTypeIfNeeded(sem->Type());
|
auto type_id = GenerateTypeIfNeeded(target->ReturnType());
|
||||||
if (type_id == 0) {
|
if (type_id == 0) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -1865,31 +1866,27 @@ uint32_t Builder::GenerateCallExpression(ast::CallExpression* expr) {
|
||||||
return result_id;
|
return result_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t Builder::GenerateIntrinsic(ast::IdentifierExpression* ident,
|
uint32_t Builder::GenerateIntrinsic(ast::CallExpression* call,
|
||||||
ast::CallExpression* call,
|
const semantic::Intrinsic* intrinsic) {
|
||||||
const semantic::IntrinsicCall* sem) {
|
|
||||||
auto result = result_op();
|
auto result = result_op();
|
||||||
auto result_id = result.to_i();
|
auto result_id = result.to_i();
|
||||||
|
|
||||||
auto result_type_id = GenerateTypeIfNeeded(TypeOf(call));
|
auto result_type_id = GenerateTypeIfNeeded(intrinsic->ReturnType());
|
||||||
if (result_type_id == 0) {
|
if (result_type_id == 0) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto intrinsic = sem->intrinsic();
|
if (intrinsic->IsFineDerivative() || intrinsic->IsCoarseDerivative()) {
|
||||||
|
|
||||||
if (semantic::intrinsic::IsFineDerivative(intrinsic) ||
|
|
||||||
semantic::intrinsic::IsCoarseDerivative(intrinsic)) {
|
|
||||||
push_capability(SpvCapabilityDerivativeControl);
|
push_capability(SpvCapabilityDerivativeControl);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (semantic::intrinsic::IsImageQueryIntrinsic(intrinsic)) {
|
if (intrinsic->IsImageQuery()) {
|
||||||
push_capability(SpvCapabilityImageQuery);
|
push_capability(SpvCapabilityImageQuery);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (auto* tex_sem = sem->As<semantic::TextureIntrinsicCall>()) {
|
if (intrinsic->IsTexture()) {
|
||||||
if (!GenerateTextureIntrinsic(ident, call, tex_sem,
|
if (!GenerateTextureIntrinsic(call, intrinsic, Operand::Int(result_type_id),
|
||||||
Operand::Int(result_type_id), result)) {
|
result)) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
return result_id;
|
return result_id;
|
||||||
|
@ -1898,97 +1895,118 @@ uint32_t Builder::GenerateIntrinsic(ast::IdentifierExpression* ident,
|
||||||
OperandList params = {Operand::Int(result_type_id), result};
|
OperandList params = {Operand::Int(result_type_id), result};
|
||||||
|
|
||||||
spv::Op op = spv::Op::OpNop;
|
spv::Op op = spv::Op::OpNop;
|
||||||
if (intrinsic == IntrinsicType::kAny) {
|
switch (intrinsic->Type()) {
|
||||||
op = spv::Op::OpAny;
|
case IntrinsicType::kAny:
|
||||||
} else if (intrinsic == IntrinsicType::kAll) {
|
op = spv::Op::OpAny;
|
||||||
op = spv::Op::OpAll;
|
break;
|
||||||
} else if (intrinsic == IntrinsicType::kArrayLength) {
|
case IntrinsicType::kAll:
|
||||||
if (call->params().empty()) {
|
op = spv::Op::OpAll;
|
||||||
error_ = "missing param for runtime array length";
|
break;
|
||||||
return 0;
|
case IntrinsicType::kArrayLength: {
|
||||||
}
|
if (call->params().empty()) {
|
||||||
auto* arg = call->params()[0];
|
error_ = "missing param for runtime array length";
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
auto* arg = call->params()[0];
|
||||||
|
|
||||||
auto* accessor = arg->As<ast::MemberAccessorExpression>();
|
auto* accessor = arg->As<ast::MemberAccessorExpression>();
|
||||||
if (accessor == nullptr) {
|
if (accessor == nullptr) {
|
||||||
error_ = "invalid expression for array length";
|
error_ = "invalid expression for array length";
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto struct_id = GenerateExpression(accessor->structure());
|
auto struct_id = GenerateExpression(accessor->structure());
|
||||||
if (struct_id == 0) {
|
if (struct_id == 0) {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
params.push_back(Operand::Int(struct_id));
|
params.push_back(Operand::Int(struct_id));
|
||||||
|
|
||||||
auto* type = TypeOf(accessor->structure())->UnwrapAll();
|
auto* type = TypeOf(accessor->structure())->UnwrapAll();
|
||||||
if (!type->Is<type::Struct>()) {
|
if (!type->Is<type::Struct>()) {
|
||||||
error_ =
|
error_ =
|
||||||
"invalid type (" + type->type_name() + ") for runtime array length";
|
"invalid type (" + type->type_name() + ") for runtime array length";
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
// Runtime array must be the last member in the structure
|
// Runtime array must be the last member in the structure
|
||||||
params.push_back(Operand::Int(
|
params.push_back(Operand::Int(
|
||||||
uint32_t(type->As<type::Struct>()->impl()->members().size() - 1)));
|
uint32_t(type->As<type::Struct>()->impl()->members().size() - 1)));
|
||||||
|
|
||||||
if (!push_function_inst(spv::Op::OpArrayLength, params)) {
|
if (!push_function_inst(spv::Op::OpArrayLength, params)) {
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
return result_id;
|
||||||
}
|
}
|
||||||
return result_id;
|
case IntrinsicType::kCountOneBits:
|
||||||
} else if (intrinsic == IntrinsicType::kCountOneBits) {
|
op = spv::Op::OpBitCount;
|
||||||
op = spv::Op::OpBitCount;
|
break;
|
||||||
} else if (intrinsic == IntrinsicType::kDot) {
|
case IntrinsicType::kDot:
|
||||||
op = spv::Op::OpDot;
|
op = spv::Op::OpDot;
|
||||||
} else if (intrinsic == IntrinsicType::kDpdx) {
|
break;
|
||||||
op = spv::Op::OpDPdx;
|
case IntrinsicType::kDpdx:
|
||||||
} else if (intrinsic == IntrinsicType::kDpdxCoarse) {
|
op = spv::Op::OpDPdx;
|
||||||
op = spv::Op::OpDPdxCoarse;
|
break;
|
||||||
} else if (intrinsic == IntrinsicType::kDpdxFine) {
|
case IntrinsicType::kDpdxCoarse:
|
||||||
op = spv::Op::OpDPdxFine;
|
op = spv::Op::OpDPdxCoarse;
|
||||||
} else if (intrinsic == IntrinsicType::kDpdy) {
|
break;
|
||||||
op = spv::Op::OpDPdy;
|
case IntrinsicType::kDpdxFine:
|
||||||
} else if (intrinsic == IntrinsicType::kDpdyCoarse) {
|
op = spv::Op::OpDPdxFine;
|
||||||
op = spv::Op::OpDPdyCoarse;
|
break;
|
||||||
} else if (intrinsic == IntrinsicType::kDpdyFine) {
|
case IntrinsicType::kDpdy:
|
||||||
op = spv::Op::OpDPdyFine;
|
op = spv::Op::OpDPdy;
|
||||||
} else if (intrinsic == IntrinsicType::kFwidth) {
|
break;
|
||||||
op = spv::Op::OpFwidth;
|
case IntrinsicType::kDpdyCoarse:
|
||||||
} else if (intrinsic == IntrinsicType::kFwidthCoarse) {
|
op = spv::Op::OpDPdyCoarse;
|
||||||
op = spv::Op::OpFwidthCoarse;
|
break;
|
||||||
} else if (intrinsic == IntrinsicType::kFwidthFine) {
|
case IntrinsicType::kDpdyFine:
|
||||||
op = spv::Op::OpFwidthFine;
|
op = spv::Op::OpDPdyFine;
|
||||||
} else if (intrinsic == IntrinsicType::kIsInf) {
|
break;
|
||||||
op = spv::Op::OpIsInf;
|
case IntrinsicType::kFwidth:
|
||||||
} else if (intrinsic == IntrinsicType::kIsNan) {
|
op = spv::Op::OpFwidth;
|
||||||
op = spv::Op::OpIsNan;
|
break;
|
||||||
} else if (intrinsic == IntrinsicType::kReverseBits) {
|
case IntrinsicType::kFwidthCoarse:
|
||||||
op = spv::Op::OpBitReverse;
|
op = spv::Op::OpFwidthCoarse;
|
||||||
} else if (intrinsic == IntrinsicType::kSelect) {
|
break;
|
||||||
op = spv::Op::OpSelect;
|
case IntrinsicType::kFwidthFine:
|
||||||
} else {
|
op = spv::Op::OpFwidthFine;
|
||||||
GenerateGLSLstd450Import();
|
break;
|
||||||
|
case IntrinsicType::kIsInf:
|
||||||
|
op = spv::Op::OpIsInf;
|
||||||
|
break;
|
||||||
|
case IntrinsicType::kIsNan:
|
||||||
|
op = spv::Op::OpIsNan;
|
||||||
|
break;
|
||||||
|
case IntrinsicType::kReverseBits:
|
||||||
|
op = spv::Op::OpBitReverse;
|
||||||
|
break;
|
||||||
|
case IntrinsicType::kSelect:
|
||||||
|
op = spv::Op::OpSelect;
|
||||||
|
break;
|
||||||
|
default: {
|
||||||
|
GenerateGLSLstd450Import();
|
||||||
|
|
||||||
auto set_iter = import_name_to_id_.find(kGLSLstd450);
|
auto set_iter = import_name_to_id_.find(kGLSLstd450);
|
||||||
if (set_iter == import_name_to_id_.end()) {
|
if (set_iter == import_name_to_id_.end()) {
|
||||||
error_ = std::string("unknown import ") + kGLSLstd450;
|
error_ = std::string("unknown import ") + kGLSLstd450;
|
||||||
return 0;
|
return 0;
|
||||||
|
}
|
||||||
|
auto set_id = set_iter->second;
|
||||||
|
auto inst_id = intrinsic_to_glsl_method(intrinsic);
|
||||||
|
if (inst_id == 0) {
|
||||||
|
error_ = "unknown method " + std::string(intrinsic->str());
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
params.push_back(Operand::Int(set_id));
|
||||||
|
params.push_back(Operand::Int(inst_id));
|
||||||
|
|
||||||
|
op = spv::Op::OpExtInst;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
auto set_id = set_iter->second;
|
|
||||||
auto inst_id = intrinsic_to_glsl_method(sem->Type(), sem->intrinsic());
|
|
||||||
if (inst_id == 0) {
|
|
||||||
error_ = "unknown method " + builder_.Symbols().NameFor(ident->symbol());
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
params.push_back(Operand::Int(set_id));
|
|
||||||
params.push_back(Operand::Int(inst_id));
|
|
||||||
|
|
||||||
op = spv::Op::OpExtInst;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (op == spv::Op::OpNop) {
|
if (op == spv::Op::OpNop) {
|
||||||
error_ = "unable to determine operator for: " +
|
error_ =
|
||||||
builder_.Symbols().NameFor(ident->symbol());
|
"unable to determine operator for: " + std::string(intrinsic->str());
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2009,32 +2027,46 @@ uint32_t Builder::GenerateIntrinsic(ast::IdentifierExpression* ident,
|
||||||
return result_id;
|
return result_id;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Builder::GenerateTextureIntrinsic(
|
bool Builder::GenerateTextureIntrinsic(ast::CallExpression* call,
|
||||||
ast::IdentifierExpression* ident,
|
const semantic::Intrinsic* intrinsic,
|
||||||
ast::CallExpression* call,
|
Operand result_type,
|
||||||
const semantic::TextureIntrinsicCall* sem,
|
Operand result_id) {
|
||||||
Operand result_type,
|
using Usage = semantic::Parameter::Usage;
|
||||||
Operand result_id) {
|
|
||||||
auto& pidx = sem->Params().idx;
|
|
||||||
auto const kNotUsed = semantic::TextureIntrinsicCall::Parameters::kNotUsed;
|
|
||||||
|
|
||||||
assert(pidx.texture != kNotUsed);
|
auto parameters = intrinsic->Parameters();
|
||||||
auto* texture_type =
|
auto arguments = call->params();
|
||||||
TypeOf(call->params()[pidx.texture])->UnwrapAll()->As<type::Texture>();
|
|
||||||
|
|
||||||
auto op = spv::Op::OpNop;
|
// Generates the given expression, returning the operand ID
|
||||||
|
auto gen = [&](ast::Expression* expr) {
|
||||||
auto gen_param = [&](size_t idx) {
|
auto val_id = GenerateExpression(expr);
|
||||||
auto* p = call->params()[idx];
|
|
||||||
auto val_id = GenerateExpression(p);
|
|
||||||
if (val_id == 0) {
|
if (val_id == 0) {
|
||||||
return Operand::Int(0);
|
return Operand::Int(0);
|
||||||
}
|
}
|
||||||
val_id = GenerateLoadIfNeeded(TypeOf(p), val_id);
|
val_id = GenerateLoadIfNeeded(TypeOf(expr), val_id);
|
||||||
|
|
||||||
return Operand::Int(val_id);
|
return Operand::Int(val_id);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// Returns the argument with the given usage
|
||||||
|
auto arg = [&](Usage usage) {
|
||||||
|
int idx = semantic::IndexOf(parameters, usage);
|
||||||
|
return (idx >= 0) ? arguments[idx] : nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Generates the argument with the given usage, returning the operand ID
|
||||||
|
auto gen_arg = [&](Usage usage) {
|
||||||
|
auto* argument = arg(usage);
|
||||||
|
assert(argument);
|
||||||
|
return gen(argument);
|
||||||
|
};
|
||||||
|
|
||||||
|
auto* texture = arg(Usage::kTexture);
|
||||||
|
assert(texture);
|
||||||
|
|
||||||
|
auto* texture_type = TypeOf(texture)->UnwrapAll()->As<type::Texture>();
|
||||||
|
|
||||||
|
auto op = spv::Op::OpNop;
|
||||||
|
|
||||||
// Custom function to call after the texture-intrinsic op has been generated.
|
// Custom function to call after the texture-intrinsic op has been generated.
|
||||||
std::function<bool()> post_emission = [] { return true; };
|
std::function<bool()> post_emission = [] { return true; };
|
||||||
|
|
||||||
|
@ -2133,28 +2165,23 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
};
|
};
|
||||||
|
|
||||||
auto append_coords_to_spirv_params = [&]() -> bool {
|
auto append_coords_to_spirv_params = [&]() -> bool {
|
||||||
if (pidx.array_index != kNotUsed) {
|
if (auto* array_index = arg(Usage::kArrayIndex)) {
|
||||||
// Array index needs to be appended to the coordinates.
|
// Array index needs to be appended to the coordinates.
|
||||||
auto* param_coords = call->params()[pidx.coords];
|
auto* packed = AppendVector(&builder_, arg(Usage::kCoords), array_index);
|
||||||
auto* param_array_index = call->params()[pidx.array_index];
|
|
||||||
|
|
||||||
auto* packed = AppendVector(&builder_, param_coords, param_array_index);
|
|
||||||
auto param = GenerateTypeConstructorExpression(packed, false);
|
auto param = GenerateTypeConstructorExpression(packed, false);
|
||||||
if (param == 0) {
|
if (param == 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
spirv_params.emplace_back(Operand::Int(param));
|
spirv_params.emplace_back(Operand::Int(param));
|
||||||
} else {
|
} else {
|
||||||
spirv_params.emplace_back(gen_param(pidx.coords)); // coordinates
|
spirv_params.emplace_back(gen_arg(Usage::kCoords)); // coordinates
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
auto append_image_and_coords_to_spirv_params = [&]() -> bool {
|
auto append_image_and_coords_to_spirv_params = [&]() -> bool {
|
||||||
assert(pidx.sampler != kNotUsed);
|
auto sampler_param = gen_arg(Usage::kSampler);
|
||||||
assert(pidx.texture != kNotUsed);
|
auto texture_param = gen_arg(Usage::kTexture);
|
||||||
auto sampler_param = gen_param(pidx.sampler);
|
|
||||||
auto texture_param = gen_param(pidx.texture);
|
|
||||||
auto sampled_image =
|
auto sampled_image =
|
||||||
GenerateSampledImage(texture_type, texture_param, sampler_param);
|
GenerateSampledImage(texture_type, texture_param, sampler_param);
|
||||||
|
|
||||||
|
@ -2163,7 +2190,7 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
return append_coords_to_spirv_params();
|
return append_coords_to_spirv_params();
|
||||||
};
|
};
|
||||||
|
|
||||||
switch (sem->intrinsic()) {
|
switch (intrinsic->Type()) {
|
||||||
case IntrinsicType::kTextureDimensions: {
|
case IntrinsicType::kTextureDimensions: {
|
||||||
// Number of returned elements from OpImageQuerySize[Lod] may not match
|
// Number of returned elements from OpImageQuerySize[Lod] may not match
|
||||||
// those of textureDimensions().
|
// those of textureDimensions().
|
||||||
|
@ -2205,13 +2232,13 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
spirv_params.emplace_back(gen_param(pidx.texture));
|
spirv_params.emplace_back(gen_arg(Usage::kTexture));
|
||||||
if (texture_type->Is<type::MultisampledTexture>() ||
|
if (texture_type->Is<type::MultisampledTexture>() ||
|
||||||
texture_type->Is<type::StorageTexture>()) {
|
texture_type->Is<type::StorageTexture>()) {
|
||||||
op = spv::Op::OpImageQuerySize;
|
op = spv::Op::OpImageQuerySize;
|
||||||
} else if (pidx.level != kNotUsed) {
|
} else if (auto* level = arg(Usage::kLevel)) {
|
||||||
op = spv::Op::OpImageQuerySizeLod;
|
op = spv::Op::OpImageQuerySizeLod;
|
||||||
spirv_params.emplace_back(gen_param(pidx.level));
|
spirv_params.emplace_back(gen(level));
|
||||||
} else {
|
} else {
|
||||||
ast::SintLiteral i32_0(Source{}, builder_.create<type::I32>(), 0);
|
ast::SintLiteral i32_0(Source{}, builder_.create<type::I32>(), 0);
|
||||||
op = spv::Op::OpImageQuerySizeLod;
|
op = spv::Op::OpImageQuerySizeLod;
|
||||||
|
@ -2242,7 +2269,7 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
spirv_params.emplace_back(gen_param(pidx.texture));
|
spirv_params.emplace_back(gen_arg(Usage::kTexture));
|
||||||
|
|
||||||
if (texture_type->Is<type::MultisampledTexture>() ||
|
if (texture_type->Is<type::MultisampledTexture>() ||
|
||||||
texture_type->Is<type::StorageTexture>()) {
|
texture_type->Is<type::StorageTexture>()) {
|
||||||
|
@ -2258,43 +2285,43 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
case IntrinsicType::kTextureNumLevels: {
|
case IntrinsicType::kTextureNumLevels: {
|
||||||
op = spv::Op::OpImageQueryLevels;
|
op = spv::Op::OpImageQueryLevels;
|
||||||
append_result_type_and_id_to_spirv_params();
|
append_result_type_and_id_to_spirv_params();
|
||||||
spirv_params.emplace_back(gen_param(pidx.texture));
|
spirv_params.emplace_back(gen_arg(Usage::kTexture));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case IntrinsicType::kTextureNumSamples: {
|
case IntrinsicType::kTextureNumSamples: {
|
||||||
op = spv::Op::OpImageQuerySamples;
|
op = spv::Op::OpImageQuerySamples;
|
||||||
append_result_type_and_id_to_spirv_params();
|
append_result_type_and_id_to_spirv_params();
|
||||||
spirv_params.emplace_back(gen_param(pidx.texture));
|
spirv_params.emplace_back(gen_arg(Usage::kTexture));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case IntrinsicType::kTextureLoad: {
|
case IntrinsicType::kTextureLoad: {
|
||||||
op = texture_type->Is<type::StorageTexture>() ? spv::Op::OpImageRead
|
op = texture_type->Is<type::StorageTexture>() ? spv::Op::OpImageRead
|
||||||
: spv::Op::OpImageFetch;
|
: spv::Op::OpImageFetch;
|
||||||
append_result_type_and_id_to_spirv_params_for_read();
|
append_result_type_and_id_to_spirv_params_for_read();
|
||||||
spirv_params.emplace_back(gen_param(pidx.texture));
|
spirv_params.emplace_back(gen_arg(Usage::kTexture));
|
||||||
if (!append_coords_to_spirv_params()) {
|
if (!append_coords_to_spirv_params()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pidx.level != kNotUsed) {
|
if (auto* level = arg(Usage::kLevel)) {
|
||||||
image_operands.emplace_back(
|
image_operands.emplace_back(
|
||||||
ImageOperand{SpvImageOperandsLodMask, gen_param(pidx.level)});
|
ImageOperand{SpvImageOperandsLodMask, gen(level)});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pidx.sample_index != kNotUsed) {
|
if (auto* sample_index = arg(Usage::kSampleIndex)) {
|
||||||
image_operands.emplace_back(ImageOperand{SpvImageOperandsSampleMask,
|
image_operands.emplace_back(
|
||||||
gen_param(pidx.sample_index)});
|
ImageOperand{SpvImageOperandsSampleMask, gen(sample_index)});
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case IntrinsicType::kTextureStore: {
|
case IntrinsicType::kTextureStore: {
|
||||||
op = spv::Op::OpImageWrite;
|
op = spv::Op::OpImageWrite;
|
||||||
spirv_params.emplace_back(gen_param(pidx.texture));
|
spirv_params.emplace_back(gen_arg(Usage::kTexture));
|
||||||
if (!append_coords_to_spirv_params()) {
|
if (!append_coords_to_spirv_params()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
spirv_params.emplace_back(gen_param(pidx.value));
|
spirv_params.emplace_back(gen_arg(Usage::kValue));
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case IntrinsicType::kTextureSample: {
|
case IntrinsicType::kTextureSample: {
|
||||||
|
@ -2311,9 +2338,8 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
if (!append_image_and_coords_to_spirv_params()) {
|
if (!append_image_and_coords_to_spirv_params()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
assert(pidx.bias != kNotUsed);
|
|
||||||
image_operands.emplace_back(
|
image_operands.emplace_back(
|
||||||
ImageOperand{SpvImageOperandsBiasMask, gen_param(pidx.bias)});
|
ImageOperand{SpvImageOperandsBiasMask, gen_arg(Usage::kBias)});
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case IntrinsicType::kTextureSampleLevel: {
|
case IntrinsicType::kTextureSampleLevel: {
|
||||||
|
@ -2322,20 +2348,19 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
if (!append_image_and_coords_to_spirv_params()) {
|
if (!append_image_and_coords_to_spirv_params()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
assert(pidx.level != kNotUsed);
|
|
||||||
auto level = Operand::Int(0);
|
auto level = Operand::Int(0);
|
||||||
if (TypeOf(call->params()[pidx.level])->Is<type::I32>()) {
|
if (TypeOf(arg(Usage::kLevel))->Is<type::I32>()) {
|
||||||
// Depth textures have i32 parameters for the level, but SPIR-V expects
|
// Depth textures have i32 parameters for the level, but SPIR-V expects
|
||||||
// F32. Cast.
|
// F32. Cast.
|
||||||
auto* f32 = builder_.create<type::F32>();
|
auto* f32 = builder_.create<type::F32>();
|
||||||
ast::TypeConstructorExpression cast(Source{}, f32,
|
ast::TypeConstructorExpression cast(Source{}, f32,
|
||||||
{call->params()[pidx.level]});
|
{arg(Usage::kLevel)});
|
||||||
level = Operand::Int(GenerateExpression(&cast));
|
level = Operand::Int(GenerateExpression(&cast));
|
||||||
if (level.to_i() == 0) {
|
if (level.to_i() == 0) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
level = gen_param(pidx.level);
|
level = gen_arg(Usage::kLevel);
|
||||||
}
|
}
|
||||||
image_operands.emplace_back(ImageOperand{SpvImageOperandsLodMask, level});
|
image_operands.emplace_back(ImageOperand{SpvImageOperandsLodMask, level});
|
||||||
break;
|
break;
|
||||||
|
@ -2346,12 +2371,10 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
if (!append_image_and_coords_to_spirv_params()) {
|
if (!append_image_and_coords_to_spirv_params()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
assert(pidx.ddx != kNotUsed);
|
|
||||||
assert(pidx.ddy != kNotUsed);
|
|
||||||
image_operands.emplace_back(
|
image_operands.emplace_back(
|
||||||
ImageOperand{SpvImageOperandsGradMask, gen_param(pidx.ddx)});
|
ImageOperand{SpvImageOperandsGradMask, gen_arg(Usage::kDdx)});
|
||||||
image_operands.emplace_back(
|
image_operands.emplace_back(
|
||||||
ImageOperand{SpvImageOperandsGradMask, gen_param(pidx.ddy)});
|
ImageOperand{SpvImageOperandsGradMask, gen_arg(Usage::kDdy)});
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case IntrinsicType::kTextureSampleCompare: {
|
case IntrinsicType::kTextureSampleCompare: {
|
||||||
|
@ -2360,8 +2383,7 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
if (!append_image_and_coords_to_spirv_params()) {
|
if (!append_image_and_coords_to_spirv_params()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
assert(pidx.depth_ref != kNotUsed);
|
spirv_params.emplace_back(gen_arg(Usage::kDepthRef));
|
||||||
spirv_params.emplace_back(gen_param(pidx.depth_ref));
|
|
||||||
|
|
||||||
type::F32 f32;
|
type::F32 f32;
|
||||||
ast::FloatLiteral float_0(Source{}, &f32, 0.0);
|
ast::FloatLiteral float_0(Source{}, &f32, 0.0);
|
||||||
|
@ -2374,9 +2396,9 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
break; // unreachable
|
break; // unreachable
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pidx.offset != kNotUsed) {
|
if (auto* offset = arg(Usage::kOffset)) {
|
||||||
image_operands.emplace_back(
|
image_operands.emplace_back(
|
||||||
ImageOperand{SpvImageOperandsConstOffsetMask, gen_param(pidx.offset)});
|
ImageOperand{SpvImageOperandsConstOffsetMask, gen(offset)});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!image_operands.empty()) {
|
if (!image_operands.empty()) {
|
||||||
|
@ -2393,8 +2415,8 @@ bool Builder::GenerateTextureIntrinsic(
|
||||||
}
|
}
|
||||||
|
|
||||||
if (op == spv::Op::OpNop) {
|
if (op == spv::Op::OpNop) {
|
||||||
error_ = "unable to determine operator for: " +
|
error_ =
|
||||||
builder_.Symbols().NameFor(ident->symbol());
|
"unable to determine operator for: " + std::string(intrinsic->str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -60,8 +60,7 @@ namespace tint {
|
||||||
|
|
||||||
// Forward declarations
|
// Forward declarations
|
||||||
namespace semantic {
|
namespace semantic {
|
||||||
class TextureIntrinsicCall;
|
class Call;
|
||||||
class IntrinsicCall;
|
|
||||||
} // namespace semantic
|
} // namespace semantic
|
||||||
|
|
||||||
namespace writer {
|
namespace writer {
|
||||||
|
@ -359,25 +358,21 @@ class Builder {
|
||||||
/// @returns the expression ID on success or 0 otherwise
|
/// @returns the expression ID on success or 0 otherwise
|
||||||
uint32_t GenerateCallExpression(ast::CallExpression* expr);
|
uint32_t GenerateCallExpression(ast::CallExpression* expr);
|
||||||
/// Generates an intrinsic call
|
/// Generates an intrinsic call
|
||||||
/// @param ident the intrinsic expression
|
|
||||||
/// @param call the call expression
|
/// @param call the call expression
|
||||||
/// @param sem the semantic information for the intrinsic call
|
/// @param intrinsic the semantic information for the intrinsic
|
||||||
/// @returns the expression ID on success or 0 otherwise
|
/// @returns the expression ID on success or 0 otherwise
|
||||||
uint32_t GenerateIntrinsic(ast::IdentifierExpression* ident,
|
uint32_t GenerateIntrinsic(ast::CallExpression* call,
|
||||||
ast::CallExpression* call,
|
const semantic::Intrinsic* intrinsic);
|
||||||
const semantic::IntrinsicCall* sem);
|
|
||||||
/// Generates a texture intrinsic call. Emits an error and returns false if
|
/// Generates a texture intrinsic call. Emits an error and returns false if
|
||||||
/// we're currently outside a function.
|
/// we're currently outside a function.
|
||||||
/// @param ident the texture intrinsic
|
|
||||||
/// @param call the call expression
|
/// @param call the call expression
|
||||||
/// @param sem the semantic information for the texture intrinsic call
|
/// @param intrinsic the semantic information for the texture intrinsic
|
||||||
/// @param result_type result type operand of the texture instruction
|
/// @param result_type result type operand of the texture instruction
|
||||||
/// @param result_id result identifier operand of the texture instruction
|
/// @param result_id result identifier operand of the texture instruction
|
||||||
/// parameters
|
/// parameters
|
||||||
/// @returns true on success
|
/// @returns true on success
|
||||||
bool GenerateTextureIntrinsic(ast::IdentifierExpression* ident,
|
bool GenerateTextureIntrinsic(ast::CallExpression* call,
|
||||||
ast::CallExpression* call,
|
const semantic::Intrinsic* intrinsic,
|
||||||
const semantic::TextureIntrinsicCall* sem,
|
|
||||||
spirv::Operand result_type,
|
spirv::Operand result_type,
|
||||||
spirv::Operand result_id);
|
spirv::Operand result_id);
|
||||||
/// Generates a sampled image
|
/// Generates a sampled image
|
||||||
|
|
Loading…
Reference in New Issue