2017-04-20 18:38:20 +00:00
|
|
|
// Copyright 2017 The NXT Authors
|
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2017-07-06 18:41:13 +00:00
|
|
|
#include "backend/Texture.h"
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2017-07-06 18:41:13 +00:00
|
|
|
#include "backend/Device.h"
|
2017-07-10 17:46:05 +00:00
|
|
|
#include "common/Assert.h"
|
2017-04-20 18:38:20 +00:00
|
|
|
|
|
|
|
namespace backend {
|
|
|
|
|
2017-07-22 00:00:22 +00:00
|
|
|
uint32_t TextureFormatPixelSize(nxt::TextureFormat format) {
|
2017-04-20 18:38:20 +00:00
|
|
|
switch (format) {
|
|
|
|
case nxt::TextureFormat::R8G8B8A8Unorm:
|
2017-08-14 16:49:41 +00:00
|
|
|
case nxt::TextureFormat::R8G8B8A8Uint:
|
2017-09-21 17:12:49 +00:00
|
|
|
case nxt::TextureFormat::B8G8R8A8Unorm:
|
2017-04-20 18:38:20 +00:00
|
|
|
return 4;
|
2017-07-17 16:03:16 +00:00
|
|
|
case nxt::TextureFormat::D32FloatS8Uint:
|
|
|
|
return 8;
|
2017-07-11 00:31:47 +00:00
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-17 16:03:16 +00:00
|
|
|
bool TextureFormatHasDepth(nxt::TextureFormat format) {
|
|
|
|
switch (format) {
|
|
|
|
case nxt::TextureFormat::R8G8B8A8Unorm:
|
2017-08-14 16:49:41 +00:00
|
|
|
case nxt::TextureFormat::R8G8B8A8Uint:
|
2017-09-21 17:12:49 +00:00
|
|
|
case nxt::TextureFormat::B8G8R8A8Unorm:
|
2017-07-17 16:03:16 +00:00
|
|
|
return false;
|
|
|
|
case nxt::TextureFormat::D32FloatS8Uint:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TextureFormatHasStencil(nxt::TextureFormat format) {
|
|
|
|
switch (format) {
|
|
|
|
case nxt::TextureFormat::R8G8B8A8Unorm:
|
2017-08-14 16:49:41 +00:00
|
|
|
case nxt::TextureFormat::R8G8B8A8Uint:
|
2017-09-21 17:12:49 +00:00
|
|
|
case nxt::TextureFormat::B8G8R8A8Unorm:
|
2017-07-17 16:03:16 +00:00
|
|
|
return false;
|
|
|
|
case nxt::TextureFormat::D32FloatS8Uint:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-11 21:36:20 +00:00
|
|
|
bool TextureFormatHasDepthOrStencil(nxt::TextureFormat format) {
|
|
|
|
switch (format) {
|
|
|
|
case nxt::TextureFormat::R8G8B8A8Unorm:
|
2017-08-29 20:35:05 +00:00
|
|
|
case nxt::TextureFormat::R8G8B8A8Uint:
|
2017-09-21 17:12:49 +00:00
|
|
|
case nxt::TextureFormat::B8G8R8A8Unorm:
|
2017-08-11 21:36:20 +00:00
|
|
|
return false;
|
|
|
|
case nxt::TextureFormat::D32FloatS8Uint:
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-17 16:03:16 +00:00
|
|
|
|
2017-04-20 18:38:20 +00:00
|
|
|
// TextureBase
|
|
|
|
|
|
|
|
TextureBase::TextureBase(TextureBuilder* builder)
|
|
|
|
: device(builder->device), dimension(builder->dimension), format(builder->format), width(builder->width),
|
|
|
|
height(builder->height), depth(builder->depth), numMipLevels(builder->numMipLevels),
|
|
|
|
allowedUsage(builder->allowedUsage), currentUsage(builder->currentUsage) {
|
|
|
|
}
|
|
|
|
|
2017-05-10 13:30:05 +00:00
|
|
|
DeviceBase* TextureBase::GetDevice() {
|
|
|
|
return device;
|
|
|
|
}
|
|
|
|
|
2017-04-20 18:38:20 +00:00
|
|
|
nxt::TextureDimension TextureBase::GetDimension() const {
|
|
|
|
return dimension;
|
|
|
|
}
|
|
|
|
nxt::TextureFormat TextureBase::GetFormat() const {
|
|
|
|
return format;
|
|
|
|
}
|
|
|
|
uint32_t TextureBase::GetWidth() const {
|
|
|
|
return width;
|
|
|
|
}
|
|
|
|
uint32_t TextureBase::GetHeight() const {
|
|
|
|
return height;
|
|
|
|
}
|
|
|
|
uint32_t TextureBase::GetDepth() const {
|
|
|
|
return depth;
|
|
|
|
}
|
|
|
|
uint32_t TextureBase::GetNumMipLevels() const {
|
|
|
|
return numMipLevels;
|
|
|
|
}
|
|
|
|
nxt::TextureUsageBit TextureBase::GetAllowedUsage() const {
|
|
|
|
return allowedUsage;
|
|
|
|
}
|
|
|
|
nxt::TextureUsageBit TextureBase::GetUsage() const {
|
|
|
|
return currentUsage;
|
|
|
|
}
|
|
|
|
|
|
|
|
TextureViewBuilder* TextureBase::CreateTextureViewBuilder() {
|
|
|
|
return new TextureViewBuilder(device, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TextureBase::IsFrozen() const {
|
|
|
|
return frozen;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TextureBase::HasFrozenUsage(nxt::TextureUsageBit usage) const {
|
|
|
|
return frozen && (usage & allowedUsage);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TextureBase::IsUsagePossible(nxt::TextureUsageBit allowedUsage, nxt::TextureUsageBit usage) {
|
|
|
|
bool allowed = (usage & allowedUsage) == usage;
|
|
|
|
bool singleUse = nxt::HasZeroOrOneBits(usage);
|
|
|
|
return allowed && singleUse;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TextureBase::IsTransitionPossible(nxt::TextureUsageBit usage) const {
|
|
|
|
if (frozen) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-07-19 22:41:17 +00:00
|
|
|
if (currentUsage == nxt::TextureUsageBit::Present) {
|
|
|
|
return false;
|
|
|
|
}
|
2017-04-20 18:38:20 +00:00
|
|
|
return IsUsagePossible(allowedUsage, usage);
|
|
|
|
}
|
|
|
|
|
2017-06-12 22:31:10 +00:00
|
|
|
void TextureBase::UpdateUsageInternal(nxt::TextureUsageBit usage) {
|
2017-07-10 17:46:05 +00:00
|
|
|
ASSERT(IsTransitionPossible(usage));
|
2017-04-20 18:38:20 +00:00
|
|
|
currentUsage = usage;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextureBase::TransitionUsage(nxt::TextureUsageBit usage) {
|
|
|
|
if (!IsTransitionPossible(usage)) {
|
|
|
|
device->HandleError("Texture frozen or usage not allowed");
|
|
|
|
return;
|
|
|
|
}
|
2017-06-12 22:31:10 +00:00
|
|
|
TransitionUsageImpl(currentUsage, usage);
|
2017-08-30 23:15:36 +00:00
|
|
|
UpdateUsageInternal(usage);
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void TextureBase::FreezeUsage(nxt::TextureUsageBit usage) {
|
|
|
|
if (!IsTransitionPossible(usage)) {
|
|
|
|
device->HandleError("Texture frozen or usage not allowed");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
allowedUsage = usage;
|
2017-06-12 22:31:10 +00:00
|
|
|
TransitionUsageImpl(currentUsage, usage);
|
2017-08-30 23:15:36 +00:00
|
|
|
UpdateUsageInternal(usage);
|
2017-04-20 18:38:20 +00:00
|
|
|
frozen = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TextureBuilder
|
|
|
|
|
|
|
|
enum TextureSetProperties {
|
|
|
|
TEXTURE_PROPERTY_DIMENSION = 0x1,
|
|
|
|
TEXTURE_PROPERTY_EXTENT = 0x2,
|
|
|
|
TEXTURE_PROPERTY_FORMAT = 0x4,
|
|
|
|
TEXTURE_PROPERTY_MIP_LEVELS = 0x8,
|
|
|
|
TEXTURE_PROPERTY_ALLOWED_USAGE = 0x10,
|
|
|
|
TEXTURE_PROPERTY_INITIAL_USAGE = 0x20,
|
|
|
|
};
|
|
|
|
|
|
|
|
TextureBuilder::TextureBuilder(DeviceBase* device)
|
2017-05-08 08:52:11 +00:00
|
|
|
: Builder(device) {
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
2017-05-08 13:17:44 +00:00
|
|
|
TextureBase* TextureBuilder::GetResultImpl() {
|
2017-04-20 18:38:20 +00:00
|
|
|
constexpr int allProperties = TEXTURE_PROPERTY_DIMENSION | TEXTURE_PROPERTY_EXTENT |
|
|
|
|
TEXTURE_PROPERTY_FORMAT | TEXTURE_PROPERTY_MIP_LEVELS | TEXTURE_PROPERTY_ALLOWED_USAGE;
|
|
|
|
if ((propertiesSet & allProperties) != allProperties) {
|
2017-05-08 08:52:11 +00:00
|
|
|
HandleError("Texture missing properties");
|
2017-04-20 18:38:20 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!TextureBase::IsUsagePossible(allowedUsage, currentUsage)) {
|
2017-05-08 08:52:11 +00:00
|
|
|
HandleError("Initial texture usage is not allowed");
|
2017-04-20 18:38:20 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(cwallez@chromium.org): check stuff based on the dimension
|
|
|
|
|
|
|
|
return device->CreateTexture(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextureBuilder::SetDimension(nxt::TextureDimension dimension) {
|
|
|
|
if ((propertiesSet & TEXTURE_PROPERTY_DIMENSION) != 0) {
|
2017-05-08 08:52:11 +00:00
|
|
|
HandleError("Texture dimension property set multiple times");
|
2017-04-20 18:38:20 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
propertiesSet |= TEXTURE_PROPERTY_DIMENSION;
|
|
|
|
this->dimension = dimension;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextureBuilder::SetExtent(uint32_t width, uint32_t height, uint32_t depth) {
|
|
|
|
if ((propertiesSet & TEXTURE_PROPERTY_EXTENT) != 0) {
|
2017-05-08 08:52:11 +00:00
|
|
|
HandleError("Texture extent property set multiple times");
|
2017-04-20 18:38:20 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (width == 0 || height == 0 || depth == 0) {
|
2017-05-08 08:52:11 +00:00
|
|
|
HandleError("Cannot create an empty texture");
|
2017-04-20 18:38:20 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
propertiesSet |= TEXTURE_PROPERTY_EXTENT;
|
|
|
|
this->width = width;
|
|
|
|
this->height = height;
|
|
|
|
this->depth = depth;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextureBuilder::SetFormat(nxt::TextureFormat format) {
|
|
|
|
if ((propertiesSet & TEXTURE_PROPERTY_FORMAT) != 0) {
|
2017-05-08 08:52:11 +00:00
|
|
|
HandleError("Texture format property set multiple times");
|
2017-04-20 18:38:20 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
propertiesSet |= TEXTURE_PROPERTY_FORMAT;
|
|
|
|
this->format = format;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextureBuilder::SetMipLevels(uint32_t numMipLevels) {
|
|
|
|
if ((propertiesSet & TEXTURE_PROPERTY_MIP_LEVELS) != 0) {
|
2017-05-08 08:52:11 +00:00
|
|
|
HandleError("Texture mip levels property set multiple times");
|
2017-04-20 18:38:20 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
propertiesSet |= TEXTURE_PROPERTY_MIP_LEVELS;
|
|
|
|
this->numMipLevels = numMipLevels;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextureBuilder::SetAllowedUsage(nxt::TextureUsageBit usage) {
|
|
|
|
if ((propertiesSet & TEXTURE_PROPERTY_ALLOWED_USAGE) != 0) {
|
2017-05-08 08:52:11 +00:00
|
|
|
HandleError("Texture allowed usage property set multiple times");
|
2017-04-20 18:38:20 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
propertiesSet |= TEXTURE_PROPERTY_ALLOWED_USAGE;
|
|
|
|
this->allowedUsage = usage;
|
|
|
|
}
|
|
|
|
|
|
|
|
void TextureBuilder::SetInitialUsage(nxt::TextureUsageBit usage) {
|
|
|
|
if ((propertiesSet & TEXTURE_PROPERTY_INITIAL_USAGE) != 0) {
|
2017-05-08 08:52:11 +00:00
|
|
|
HandleError("Texture initial usage property set multiple times");
|
2017-04-20 18:38:20 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
propertiesSet |= TEXTURE_PROPERTY_INITIAL_USAGE;
|
|
|
|
this->currentUsage = usage;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TextureViewBase
|
|
|
|
|
|
|
|
TextureViewBase::TextureViewBase(TextureViewBuilder* builder)
|
|
|
|
: texture(builder->texture) {
|
|
|
|
}
|
|
|
|
|
|
|
|
TextureBase* TextureViewBase::GetTexture() {
|
|
|
|
return texture.Get();
|
|
|
|
}
|
|
|
|
|
|
|
|
// TextureViewBuilder
|
|
|
|
|
|
|
|
TextureViewBuilder::TextureViewBuilder(DeviceBase* device, TextureBase* texture)
|
2017-05-08 08:52:11 +00:00
|
|
|
: Builder(device), texture(texture) {
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
2017-05-08 13:17:44 +00:00
|
|
|
TextureViewBase* TextureViewBuilder::GetResultImpl() {
|
2017-04-20 18:38:20 +00:00
|
|
|
return device->CreateTextureView(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|