blob: 88b9957832e173177ad6981872a486e17956ba04 [file] [log] [blame]
// Copyright 2025 The Dawn & Tint Authors
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// 1. Redistributions of source code must retain the above copyright notice, this
// list of conditions and the following disclaimer.
//
// 2. Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
//
// 3. Neither the name of the copyright holder nor the names of its
// contributors may be used to endorse or promote products derived from
// this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
// DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
// FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
// DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
// SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
// OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
#ifndef SRC_DAWN_NATIVE_IMMEDIATECONSTANTSLAYOUT_H_
#define SRC_DAWN_NATIVE_IMMEDIATECONSTANTSLAYOUT_H_
#include "dawn/common/Compiler.h"
#include "dawn/common/ityp_bitset.h"
#include "dawn/native/EnumClassBitmasks.h"
#include "dawn/native/IntegerTypes.h"
namespace dawn::native {
// Define common immediate data layout. Append members to expand layouts.
// NOTE: 'offsetof' doesn't support non-standard-layout structs. So use
// aggregate instead of inheritance for RenderImmediateConstants and
// ComputeImmediateConstants.
DAWN_ENABLE_STRUCT_PADDING_WARNINGS
struct UserImmediateConstants {
uint32_t userImmediateData[kMaxExternalImmediateConstantsPerPipeline];
};
// 8 bytes of immediate data data to be used by the ClampFragDepth Tint transform.
struct ClampFragDepthArgs {
float minClampFragDepth;
float maxClampFragDepth;
};
// Define render pipeline immediate data layout. Append members to
// expand the layout.
struct RenderImmediateConstants {
UserImmediateConstants userConstants;
ClampFragDepthArgs clampFragDepth;
// first index offset
uint32_t firstVertex;
uint32_t firstInstance;
};
struct NumWorkgroupsDimensions {
uint32_t numWorkgroupsX;
uint32_t numWorkgroupsY;
uint32_t numWorkgroupsZ;
};
// Define compute pipeline immediate data layout. Append members to
// expand the layout.
struct ComputeImmediateConstants {
UserImmediateConstants userConstants;
NumWorkgroupsDimensions numWorkgroups;
};
DAWN_DISABLE_STRUCT_PADDING_WARNINGS
// Convert byte sizes and offsets into immediate constant indices and offsets
// (dividing everything by kImmediateConstantElementByteSize)
constexpr ImmediateConstantMask GetImmediateConstantBlockBits(size_t byteOffset, size_t byteSize) {
// This bit math can be done in uint64_t because there are <= 64 bits in the mask.
static_assert(ImmediateConstantMask{}.size() <= 64);
uint64_t firstIndex = byteOffset / kImmediateConstantElementByteSize;
uint64_t constantCount = byteSize / kImmediateConstantElementByteSize;
return ((1u << constantCount) - 1u) << firstIndex;
}
// Returns the offset of the member in the packed immediates of the pipeline.
// The pointer-to-member is a pointer into the structure containing all the potential immediates.
// However pipelines don't need all of them and use a compacted layout with immediates
// in the same order, just some of them skipped. For example the pipeline mask 11001111,
// representing "userConstants: 4 | trivial_constants: 0 (2 at most)|clamp_frag:2",
// maps to pipeline immediate constant layout: "userConstants:4 | clamp_frag:2
template <typename Object, typename Member>
uint32_t GetImmediateByteOffsetInPipeline(Member Object::*ptr,
const ImmediateConstantMask& pipelineImmediateMask) {
Object obj = {};
ptrdiff_t offset = reinterpret_cast<char*>(&(obj.*ptr)) - reinterpret_cast<char*>(&obj);
const ImmediateConstantMask prefixBits =
(1u << (offset / kImmediateConstantElementByteSize)) - 1u;
return (prefixBits & pipelineImmediateMask).count() * kImmediateConstantElementByteSize;
}
template <typename Object, typename Member>
bool HasImmediateConstants(Member Object::*ptr,
const ImmediateConstantMask& pipelineImmediateMask) {
Object obj = {};
ptrdiff_t offset = reinterpret_cast<char*>(&(obj.*ptr)) - reinterpret_cast<char*>(&obj);
size_t size = sizeof(Member);
return pipelineImmediateMask.to_ulong() &
GetImmediateConstantBlockBits(offset, size).to_ulong();
}
template <typename Object, typename Member>
std::optional<uint32_t> GetImmediateByteOffsetInPipelineIfAny(
Member Object::* ptr,
const ImmediateConstantMask& pipelineImmediateMask) {
if (!HasImmediateConstants(ptr, pipelineImmediateMask)) {
return std::nullopt;
}
return GetImmediateByteOffsetInPipeline(ptr, pipelineImmediateMask);
}
uint32_t GetImmediateIndexInPipeline(const uint32_t layoutOffset,
const ImmediateConstantMask& pipelineImmediateMask);
} // namespace dawn::native
#endif // SRC_DAWN_NATIVE_IMMEDIATECONSTANTSLAYOUT_H_