blob: e8f6b14c5ee225614caf9e136d018e68cf5215ba [file] [log] [blame]
// Copyright 2016 The SwiftShader Authors. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#ifndef sw_VertexProcessor_hpp
#define sw_VertexProcessor_hpp
#include "Context.hpp"
#include "Memset.hpp"
#include "RoutineCache.hpp"
#include "Vertex.hpp"
#include "Pipeline/SpirvShader.hpp"
#include <memory>
namespace sw {
struct DrawData;
// Basic direct mapped vertex cache.
struct VertexCache
{
static constexpr uint32_t SIZE = 64; // TODO: Variable size?
static constexpr uint32_t TAG_MASK = SIZE - 1; // Size must be power of 2.
void clear();
Vertex vertex[SIZE];
uint32_t tag[SIZE];
// Identifier of the draw call for the cache data. If this cache is
// used with a different draw call, then the cache should be invalidated
// before use.
int drawCall = -1;
};
struct VertexTask
{
unsigned int vertexCount;
unsigned int primitiveStart;
VertexCache vertexCache;
};
using VertexRoutineFunction = FunctionT<void(const vk::Device *device, Vertex *output, unsigned int *batch, VertexTask *vertextask, DrawData *draw)>;
class VertexProcessor
{
public:
struct States : Memset<States>
{
States()
: Memset(this, 0)
{}
uint32_t computeHash();
uint64_t shaderID;
uint32_t pipelineLayoutIdentifier;
struct Input
{
operator bool() const // Returns true if stream contains data
{
return format != VK_FORMAT_UNDEFINED;
}
VkFormat format; // TODO(b/148016460): Could be restricted to VK_FORMAT_END_RANGE
unsigned int attribType : BITS(SpirvShader::ATTRIBTYPE_LAST);
};
Input input[MAX_INTERFACE_COMPONENTS / 4];
bool robustBufferAccess : 1;
bool isPoint : 1;
bool depthClipEnable : 1;
};
struct State : States
{
bool operator==(const State &state) const;
uint32_t hash;
};
using RoutineType = VertexRoutineFunction::RoutineType;
VertexProcessor();
const State update(const vk::GraphicsState &pipelineState, const sw::SpirvShader *vertexShader, const vk::Inputs &inputs);
RoutineType routine(const State &state, vk::PipelineLayout const *pipelineLayout,
SpirvShader const *vertexShader, const vk::DescriptorSet::Bindings &descriptorSets);
void setRoutineCacheSize(int cacheSize);
private:
using RoutineCacheType = RoutineCache<State, VertexRoutineFunction::CFunctionType>;
std::unique_ptr<RoutineCacheType> routineCache;
};
} // namespace sw
namespace std {
template<>
struct hash<sw::VertexProcessor::State>
{
uint64_t operator()(const sw::VertexProcessor::State &state) const
{
return state.hash;
}
};
} // namespace std
#endif // sw_VertexProcessor_hpp