blob: 5c663093b3c16e0d3682e6ffd25d91381b4ee07b [file] [log] [blame]
Nicolas Capens68a82382018-10-02 13:16:55 -04001// Copyright 2016 The SwiftShader Authors. All Rights Reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15#include "VertexProcessor.hpp"
16
Nicolas Capens1d8c8db2018-11-05 16:30:42 -050017#include "Pipeline/VertexProgram.hpp"
Nicolas Capens1d8c8db2018-11-05 16:30:42 -050018#include "Pipeline/Constants.hpp"
19#include "System/Math.hpp"
Chris Forbesebe5f7f2019-01-16 10:38:34 -080020#include "Vulkan/VkDebug.hpp"
Nicolas Capens68a82382018-10-02 13:16:55 -040021
22#include <string.h>
23
24namespace sw
25{
Nicolas Capens68a82382018-10-02 13:16:55 -040026 void VertexCache::clear()
27 {
Nicolas Capens4a105592018-01-02 23:41:25 -050028 for(uint32_t i = 0; i < SIZE; i++)
Nicolas Capens68a82382018-10-02 13:16:55 -040029 {
Nicolas Capens4a105592018-01-02 23:41:25 -050030 tag[i] = 0xFFFFFFFF;
Nicolas Capens68a82382018-10-02 13:16:55 -040031 }
32 }
33
34 unsigned int VertexProcessor::States::computeHash()
35 {
36 unsigned int *state = (unsigned int*)this;
37 unsigned int hash = 0;
38
39 for(unsigned int i = 0; i < sizeof(States) / 4; i++)
40 {
41 hash ^= state[i];
42 }
43
44 return hash;
45 }
46
47 VertexProcessor::State::State()
48 {
49 memset(this, 0, sizeof(State));
50 }
51
52 bool VertexProcessor::State::operator==(const State &state) const
53 {
54 if(hash != state.hash)
55 {
56 return false;
57 }
58
59 return memcmp(static_cast<const States*>(this), static_cast<const States*>(&state), sizeof(States)) == 0;
60 }
61
Alexis Hetu2e4f6e82019-05-08 15:52:21 -040062 VertexProcessor::VertexProcessor()
Nicolas Capens68a82382018-10-02 13:16:55 -040063 {
Nicolas Capensdc2966a2018-10-31 11:25:15 -040064 routineCache = nullptr;
Nicolas Capens68a82382018-10-02 13:16:55 -040065 setRoutineCacheSize(1024);
66 }
67
68 VertexProcessor::~VertexProcessor()
69 {
70 delete routineCache;
Nicolas Capensdc2966a2018-10-31 11:25:15 -040071 routineCache = nullptr;
Nicolas Capens68a82382018-10-02 13:16:55 -040072 }
73
Nicolas Capens68a82382018-10-02 13:16:55 -040074 void VertexProcessor::setRoutineCacheSize(int cacheSize)
75 {
76 delete routineCache;
Alexis Hetuf2878342019-03-12 16:32:04 -040077 routineCache = new RoutineCache<State>(clamp(cacheSize, 1, 65536));
Nicolas Capens68a82382018-10-02 13:16:55 -040078 }
79
Alexis Hetu2e4f6e82019-05-08 15:52:21 -040080 const VertexProcessor::State VertexProcessor::update(const sw::Context* context)
Nicolas Capens68a82382018-10-02 13:16:55 -040081 {
Nicolas Capens68a82382018-10-02 13:16:55 -040082 State state;
83
Nicolas Capensdc2966a2018-10-31 11:25:15 -040084 state.shaderID = context->vertexShader->getSerialID();
Nicolas Capens68a82382018-10-02 13:16:55 -040085
Chris Forbes8a718cb2019-02-21 10:20:26 -080086 for(int i = 0; i < MAX_VERTEX_INPUTS; i++)
87 {
88 state.input[i].type = context->input[i].type;
89 state.input[i].count = context->input[i].count;
90 state.input[i].normalized = context->input[i].normalized;
91 // TODO: get rid of attribType -- just keep the VK format all the way through, this fully determines
92 // how to handle the attribute.
93 state.input[i].attribType = context->vertexShader->inputs[i*4].Type;
94 }
95
Nicolas Capens68a82382018-10-02 13:16:55 -040096 state.hash = state.computeHash();
97
98 return state;
99 }
100
Alexis Hetu2e4f6e82019-05-08 15:52:21 -0400101 Routine *VertexProcessor::routine(const State &state,
102 vk::PipelineLayout const *pipelineLayout,
103 SpirvShader const *vertexShader,
104 const vk::DescriptorSet::Bindings &descriptorSets)
Nicolas Capens68a82382018-10-02 13:16:55 -0400105 {
106 Routine *routine = routineCache->query(state);
107
108 if(!routine) // Create one
109 {
Alexis Hetu2e4f6e82019-05-08 15:52:21 -0400110 VertexRoutine *generator = new VertexProgram(state, pipelineLayout, vertexShader, descriptorSets);
Nicolas Capens68a82382018-10-02 13:16:55 -0400111 generator->generate();
Chris Forbes878d4b02019-01-21 10:48:35 -0800112 routine = (*generator)("VertexRoutine_%0.8X", state.shaderID);
Nicolas Capens68a82382018-10-02 13:16:55 -0400113 delete generator;
114
115 routineCache->add(state, routine);
116 }
117
118 return routine;
119 }
120}