2018-07-18 09:40:26 +00:00
|
|
|
// Copyright 2017 The Dawn Authors
|
2017-04-20 18:38:20 +00:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2018-07-24 11:53:51 +00:00
|
|
|
#include "dawn_native/Queue.h"
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2020-08-20 11:25:49 +00:00
|
|
|
#include "common/Constants.h"
|
2018-11-07 10:02:43 +00:00
|
|
|
#include "dawn_native/Buffer.h"
|
2018-07-24 11:53:51 +00:00
|
|
|
#include "dawn_native/CommandBuffer.h"
|
2020-07-08 18:42:30 +00:00
|
|
|
#include "dawn_native/CommandValidation.h"
|
2020-08-20 13:29:39 +00:00
|
|
|
#include "dawn_native/Commands.h"
|
2018-07-24 11:53:51 +00:00
|
|
|
#include "dawn_native/Device.h"
|
2020-06-02 09:24:39 +00:00
|
|
|
#include "dawn_native/DynamicUploader.h"
|
2019-09-17 18:24:07 +00:00
|
|
|
#include "dawn_native/ErrorScope.h"
|
|
|
|
#include "dawn_native/ErrorScopeTracker.h"
|
2018-12-03 16:57:34 +00:00
|
|
|
#include "dawn_native/Fence.h"
|
|
|
|
#include "dawn_native/FenceSignalTracker.h"
|
2020-07-01 10:48:16 +00:00
|
|
|
#include "dawn_native/QuerySet.h"
|
2018-11-07 10:02:43 +00:00
|
|
|
#include "dawn_native/Texture.h"
|
2019-10-28 23:15:40 +00:00
|
|
|
#include "dawn_platform/DawnPlatform.h"
|
2019-08-13 19:00:34 +00:00
|
|
|
#include "dawn_platform/tracing/TraceEvent.h"
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2020-06-03 13:36:15 +00:00
|
|
|
#include <cstring>
|
|
|
|
|
2018-07-24 14:45:45 +00:00
|
|
|
namespace dawn_native {
|
2020-08-20 13:29:39 +00:00
|
|
|
namespace {
|
|
|
|
void CopyTextureData(uint8_t* dstPointer,
|
|
|
|
const uint8_t* srcPointer,
|
|
|
|
uint32_t depth,
|
|
|
|
uint32_t rowsPerImageInBlock,
|
|
|
|
uint64_t imageAdditionalStride,
|
|
|
|
uint32_t actualBytesPerRow,
|
|
|
|
uint32_t dstBytesPerRow,
|
|
|
|
uint32_t srcBytesPerRow) {
|
|
|
|
bool copyWholeLayer =
|
|
|
|
actualBytesPerRow == dstBytesPerRow && dstBytesPerRow == srcBytesPerRow;
|
|
|
|
bool copyWholeData = copyWholeLayer && imageAdditionalStride == 0;
|
2017-04-20 18:38:20 +00:00
|
|
|
|
2020-08-20 13:29:39 +00:00
|
|
|
if (!copyWholeLayer) { // copy row by row
|
|
|
|
for (uint32_t d = 0; d < depth; ++d) {
|
|
|
|
for (uint32_t h = 0; h < rowsPerImageInBlock; ++h) {
|
|
|
|
memcpy(dstPointer, srcPointer, actualBytesPerRow);
|
|
|
|
dstPointer += dstBytesPerRow;
|
|
|
|
srcPointer += srcBytesPerRow;
|
|
|
|
}
|
|
|
|
srcPointer += imageAdditionalStride;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
uint64_t layerSize = uint64_t(rowsPerImageInBlock) * actualBytesPerRow;
|
|
|
|
if (!copyWholeData) { // copy layer by layer
|
|
|
|
for (uint32_t d = 0; d < depth; ++d) {
|
|
|
|
memcpy(dstPointer, srcPointer, layerSize);
|
|
|
|
dstPointer += layerSize;
|
|
|
|
srcPointer += layerSize + imageAdditionalStride;
|
|
|
|
}
|
|
|
|
} else { // do a single copy
|
|
|
|
memcpy(dstPointer, srcPointer, layerSize * depth);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ResultOrError<UploadHandle> UploadTextureDataAligningBytesPerRowAndOffset(
|
|
|
|
DeviceBase* device,
|
|
|
|
const void* data,
|
|
|
|
uint32_t alignedBytesPerRow,
|
|
|
|
uint32_t optimallyAlignedBytesPerRow,
|
|
|
|
uint32_t alignedRowsPerImage,
|
|
|
|
const TextureDataLayout& dataLayout,
|
|
|
|
const TexelBlockInfo& blockInfo,
|
|
|
|
const Extent3D& writeSizePixel) {
|
|
|
|
uint64_t newDataSizeBytes;
|
|
|
|
DAWN_TRY_ASSIGN(
|
|
|
|
newDataSizeBytes,
|
|
|
|
ComputeRequiredBytesInCopy(blockInfo, writeSizePixel, optimallyAlignedBytesPerRow,
|
|
|
|
alignedRowsPerImage));
|
|
|
|
|
|
|
|
uint64_t optimalOffsetAlignment =
|
|
|
|
device->GetOptimalBufferToTextureCopyOffsetAlignment();
|
|
|
|
ASSERT(IsPowerOfTwo(optimalOffsetAlignment));
|
|
|
|
ASSERT(IsPowerOfTwo(blockInfo.blockByteSize));
|
|
|
|
// We need the offset to be aligned to both optimalOffsetAlignment and blockByteSize,
|
|
|
|
// since both of them are powers of two, we only need to align to the max value.
|
|
|
|
uint64_t offsetAlignment =
|
|
|
|
std::max(optimalOffsetAlignment, uint64_t(blockInfo.blockByteSize));
|
|
|
|
|
|
|
|
UploadHandle uploadHandle;
|
|
|
|
DAWN_TRY_ASSIGN(uploadHandle, device->GetDynamicUploader()->Allocate(
|
|
|
|
newDataSizeBytes, device->GetPendingCommandSerial(),
|
|
|
|
offsetAlignment));
|
|
|
|
ASSERT(uploadHandle.mappedBuffer != nullptr);
|
|
|
|
|
|
|
|
uint8_t* dstPointer = static_cast<uint8_t*>(uploadHandle.mappedBuffer);
|
|
|
|
const uint8_t* srcPointer = static_cast<const uint8_t*>(data);
|
|
|
|
srcPointer += dataLayout.offset;
|
|
|
|
|
|
|
|
uint32_t alignedRowsPerImageInBlock = alignedRowsPerImage / blockInfo.blockHeight;
|
|
|
|
uint32_t dataRowsPerImageInBlock = dataLayout.rowsPerImage / blockInfo.blockHeight;
|
|
|
|
if (dataRowsPerImageInBlock == 0) {
|
|
|
|
dataRowsPerImageInBlock = writeSizePixel.height / blockInfo.blockHeight;
|
|
|
|
}
|
|
|
|
|
|
|
|
ASSERT(dataRowsPerImageInBlock >= alignedRowsPerImageInBlock);
|
|
|
|
uint64_t imageAdditionalStride =
|
|
|
|
dataLayout.bytesPerRow * (dataRowsPerImageInBlock - alignedRowsPerImageInBlock);
|
|
|
|
|
|
|
|
CopyTextureData(dstPointer, srcPointer, writeSizePixel.depth,
|
|
|
|
alignedRowsPerImageInBlock, imageAdditionalStride, alignedBytesPerRow,
|
|
|
|
optimallyAlignedBytesPerRow, dataLayout.bytesPerRow);
|
|
|
|
|
|
|
|
return uploadHandle;
|
|
|
|
}
|
2020-09-24 14:56:50 +00:00
|
|
|
|
|
|
|
class ErrorQueue : public QueueBase {
|
|
|
|
public:
|
|
|
|
ErrorQueue(DeviceBase* device) : QueueBase(device, ObjectBase::kError) {
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
MaybeError SubmitImpl(uint32_t commandCount,
|
|
|
|
CommandBufferBase* const* commands) override {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2020-08-20 13:29:39 +00:00
|
|
|
} // namespace
|
2020-09-24 14:56:50 +00:00
|
|
|
|
2017-04-20 18:38:20 +00:00
|
|
|
// QueueBase
|
|
|
|
|
2018-10-15 12:54:30 +00:00
|
|
|
QueueBase::QueueBase(DeviceBase* device) : ObjectBase(device) {
|
2017-05-10 13:30:05 +00:00
|
|
|
}
|
|
|
|
|
2020-01-17 21:32:08 +00:00
|
|
|
QueueBase::QueueBase(DeviceBase* device, ObjectBase::ErrorTag tag) : ObjectBase(device, tag) {
|
|
|
|
}
|
|
|
|
|
|
|
|
// static
|
|
|
|
QueueBase* QueueBase::MakeError(DeviceBase* device) {
|
2020-09-24 14:56:50 +00:00
|
|
|
return new ErrorQueue(device);
|
2020-01-17 21:32:08 +00:00
|
|
|
}
|
|
|
|
|
2019-02-21 00:45:19 +00:00
|
|
|
void QueueBase::Submit(uint32_t commandCount, CommandBufferBase* const* commands) {
|
2020-08-13 20:25:39 +00:00
|
|
|
SubmitInternal(commandCount, commands);
|
2020-01-15 19:02:13 +00:00
|
|
|
|
2020-08-13 20:25:39 +00:00
|
|
|
for (uint32_t i = 0; i < commandCount; ++i) {
|
|
|
|
commands[i]->Destroy();
|
2019-09-23 21:21:10 +00:00
|
|
|
}
|
2018-08-01 13:12:10 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 15:11:04 +00:00
|
|
|
void QueueBase::Signal(Fence* fence, uint64_t apiSignalValue) {
|
|
|
|
FenceAPISerial signalValue(apiSignalValue);
|
|
|
|
|
2019-09-17 18:24:07 +00:00
|
|
|
DeviceBase* device = GetDevice();
|
|
|
|
if (device->ConsumedError(ValidateSignal(fence, signalValue))) {
|
2018-12-03 16:57:34 +00:00
|
|
|
return;
|
|
|
|
}
|
2019-02-13 13:09:18 +00:00
|
|
|
ASSERT(!IsError());
|
2018-12-03 16:57:34 +00:00
|
|
|
|
|
|
|
fence->SetSignaledValue(signalValue);
|
2019-09-17 18:24:07 +00:00
|
|
|
device->GetFenceSignalTracker()->UpdateFenceOnComplete(fence, signalValue);
|
|
|
|
device->GetErrorScopeTracker()->TrackUntilLastSubmitComplete(
|
|
|
|
device->GetCurrentErrorScope());
|
2018-12-03 16:57:34 +00:00
|
|
|
}
|
|
|
|
|
2019-11-13 17:00:37 +00:00
|
|
|
Fence* QueueBase::CreateFence(const FenceDescriptor* descriptor) {
|
2019-03-06 22:42:22 +00:00
|
|
|
if (GetDevice()->ConsumedError(ValidateCreateFence(descriptor))) {
|
2019-11-13 17:00:37 +00:00
|
|
|
return Fence::MakeError(GetDevice());
|
2019-03-06 22:42:22 +00:00
|
|
|
}
|
|
|
|
|
2020-05-11 20:29:22 +00:00
|
|
|
if (descriptor == nullptr) {
|
|
|
|
FenceDescriptor defaultDescriptor = {};
|
|
|
|
return new Fence(this, &defaultDescriptor);
|
|
|
|
}
|
2019-11-13 17:00:37 +00:00
|
|
|
return new Fence(this, descriptor);
|
2019-03-06 22:42:22 +00:00
|
|
|
}
|
|
|
|
|
2020-06-02 09:24:39 +00:00
|
|
|
void QueueBase::WriteBuffer(BufferBase* buffer,
|
|
|
|
uint64_t bufferOffset,
|
|
|
|
const void* data,
|
|
|
|
size_t size) {
|
|
|
|
GetDevice()->ConsumedError(WriteBufferInternal(buffer, bufferOffset, data, size));
|
|
|
|
}
|
|
|
|
|
|
|
|
MaybeError QueueBase::WriteBufferInternal(BufferBase* buffer,
|
|
|
|
uint64_t bufferOffset,
|
|
|
|
const void* data,
|
|
|
|
size_t size) {
|
|
|
|
DAWN_TRY(ValidateWriteBuffer(buffer, bufferOffset, size));
|
|
|
|
return WriteBufferImpl(buffer, bufferOffset, data, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
MaybeError QueueBase::WriteBufferImpl(BufferBase* buffer,
|
|
|
|
uint64_t bufferOffset,
|
|
|
|
const void* data,
|
|
|
|
size_t size) {
|
2020-06-05 15:44:03 +00:00
|
|
|
if (size == 0) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-06-02 09:24:39 +00:00
|
|
|
DeviceBase* device = GetDevice();
|
|
|
|
|
|
|
|
UploadHandle uploadHandle;
|
|
|
|
DAWN_TRY_ASSIGN(uploadHandle, device->GetDynamicUploader()->Allocate(
|
2020-08-20 11:25:49 +00:00
|
|
|
size, device->GetPendingCommandSerial(),
|
|
|
|
kCopyBufferToBufferOffsetAlignment));
|
2020-06-02 09:24:39 +00:00
|
|
|
ASSERT(uploadHandle.mappedBuffer != nullptr);
|
|
|
|
|
|
|
|
memcpy(uploadHandle.mappedBuffer, data, size);
|
|
|
|
|
2020-06-05 15:44:03 +00:00
|
|
|
return device->CopyFromStagingToBuffer(uploadHandle.stagingBuffer, uploadHandle.startOffset,
|
|
|
|
buffer, bufferOffset, size);
|
2020-06-02 09:24:39 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 18:42:30 +00:00
|
|
|
void QueueBase::WriteTexture(const TextureCopyView* destination,
|
|
|
|
const void* data,
|
|
|
|
size_t dataSize,
|
|
|
|
const TextureDataLayout* dataLayout,
|
|
|
|
const Extent3D* writeSize) {
|
|
|
|
GetDevice()->ConsumedError(
|
|
|
|
WriteTextureInternal(destination, data, dataSize, dataLayout, writeSize));
|
|
|
|
}
|
|
|
|
|
|
|
|
MaybeError QueueBase::WriteTextureInternal(const TextureCopyView* destination,
|
|
|
|
const void* data,
|
|
|
|
size_t dataSize,
|
|
|
|
const TextureDataLayout* dataLayout,
|
|
|
|
const Extent3D* writeSize) {
|
|
|
|
DAWN_TRY(ValidateWriteTexture(destination, dataSize, dataLayout, writeSize));
|
2020-07-15 18:06:07 +00:00
|
|
|
|
|
|
|
if (writeSize->width == 0 || writeSize->height == 0 || writeSize->depth == 0) {
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-08-10 22:29:49 +00:00
|
|
|
return WriteTextureImpl(*destination, data, *dataLayout, *writeSize);
|
2020-07-08 18:42:30 +00:00
|
|
|
}
|
|
|
|
|
2020-08-10 22:29:49 +00:00
|
|
|
MaybeError QueueBase::WriteTextureImpl(const TextureCopyView& destination,
|
2020-07-08 18:42:30 +00:00
|
|
|
const void* data,
|
2020-08-10 22:29:49 +00:00
|
|
|
const TextureDataLayout& dataLayout,
|
2020-08-20 13:29:39 +00:00
|
|
|
const Extent3D& writeSizePixel) {
|
|
|
|
const TexelBlockInfo& blockInfo =
|
|
|
|
destination.texture->GetFormat().GetTexelBlockInfo(destination.aspect);
|
|
|
|
|
|
|
|
// We are only copying the part of the data that will appear in the texture.
|
|
|
|
// Note that validating texture copy range ensures that writeSizePixel->width and
|
|
|
|
// writeSizePixel->height are multiples of blockWidth and blockHeight respectively.
|
|
|
|
uint32_t alignedBytesPerRow =
|
|
|
|
(writeSizePixel.width) / blockInfo.blockWidth * blockInfo.blockByteSize;
|
|
|
|
uint32_t alignedRowsPerImage = writeSizePixel.height;
|
2020-07-08 18:42:30 +00:00
|
|
|
|
2020-08-20 13:29:39 +00:00
|
|
|
uint32_t optimalBytesPerRowAlignment = GetDevice()->GetOptimalBytesPerRowAlignment();
|
|
|
|
uint32_t optimallyAlignedBytesPerRow =
|
|
|
|
Align(alignedBytesPerRow, optimalBytesPerRowAlignment);
|
|
|
|
|
|
|
|
UploadHandle uploadHandle;
|
|
|
|
DAWN_TRY_ASSIGN(uploadHandle,
|
|
|
|
UploadTextureDataAligningBytesPerRowAndOffset(
|
|
|
|
GetDevice(), data, alignedBytesPerRow, optimallyAlignedBytesPerRow,
|
|
|
|
alignedRowsPerImage, dataLayout, blockInfo, writeSizePixel));
|
|
|
|
|
|
|
|
TextureDataLayout passDataLayout = dataLayout;
|
|
|
|
passDataLayout.offset = uploadHandle.startOffset;
|
|
|
|
passDataLayout.bytesPerRow = optimallyAlignedBytesPerRow;
|
|
|
|
passDataLayout.rowsPerImage = alignedRowsPerImage;
|
|
|
|
|
|
|
|
TextureCopy textureCopy;
|
|
|
|
textureCopy.texture = destination.texture;
|
|
|
|
textureCopy.mipLevel = destination.mipLevel;
|
|
|
|
textureCopy.origin = destination.origin;
|
|
|
|
textureCopy.aspect = ConvertAspect(destination.texture->GetFormat(), destination.aspect);
|
|
|
|
|
|
|
|
return GetDevice()->CopyFromStagingToTexture(uploadHandle.stagingBuffer, passDataLayout,
|
|
|
|
&textureCopy, writeSizePixel);
|
|
|
|
}
|
2019-02-21 00:45:19 +00:00
|
|
|
MaybeError QueueBase::ValidateSubmit(uint32_t commandCount,
|
2020-06-02 09:24:39 +00:00
|
|
|
CommandBufferBase* const* commands) const {
|
2019-10-28 23:15:40 +00:00
|
|
|
TRACE_EVENT0(GetDevice()->GetPlatform(), Validation, "Queue::ValidateSubmit");
|
2019-02-13 13:09:18 +00:00
|
|
|
DAWN_TRY(GetDevice()->ValidateObject(this));
|
|
|
|
|
2019-02-21 00:45:19 +00:00
|
|
|
for (uint32_t i = 0; i < commandCount; ++i) {
|
2019-02-14 18:13:47 +00:00
|
|
|
DAWN_TRY(GetDevice()->ValidateObject(commands[i]));
|
2020-08-13 20:25:39 +00:00
|
|
|
DAWN_TRY(commands[i]->ValidateCanUseInSubmitNow());
|
2019-02-14 18:13:47 +00:00
|
|
|
|
2018-11-07 10:02:43 +00:00
|
|
|
const CommandBufferResourceUsage& usages = commands[i]->GetResourceUsages();
|
|
|
|
|
|
|
|
for (const PassResourceUsage& passUsages : usages.perPass) {
|
|
|
|
for (const BufferBase* buffer : passUsages.buffers) {
|
2020-06-02 09:24:39 +00:00
|
|
|
DAWN_TRY(buffer->ValidateCanUseOnQueueNow());
|
2018-11-07 10:02:43 +00:00
|
|
|
}
|
|
|
|
for (const TextureBase* texture : passUsages.textures) {
|
|
|
|
DAWN_TRY(texture->ValidateCanUseInSubmitNow());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const BufferBase* buffer : usages.topLevelBuffers) {
|
2020-06-02 09:24:39 +00:00
|
|
|
DAWN_TRY(buffer->ValidateCanUseOnQueueNow());
|
2018-11-07 10:02:43 +00:00
|
|
|
}
|
|
|
|
for (const TextureBase* texture : usages.topLevelTextures) {
|
|
|
|
DAWN_TRY(texture->ValidateCanUseInSubmitNow());
|
|
|
|
}
|
2020-07-01 10:48:16 +00:00
|
|
|
for (const QuerySetBase* querySet : usages.usedQuerySets) {
|
|
|
|
DAWN_TRY(querySet->ValidateCanUseInSubmitNow());
|
|
|
|
}
|
2018-11-07 10:02:43 +00:00
|
|
|
}
|
|
|
|
|
2018-07-16 16:20:09 +00:00
|
|
|
return {};
|
2017-04-20 18:38:20 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 15:11:04 +00:00
|
|
|
MaybeError QueueBase::ValidateSignal(const Fence* fence, FenceAPISerial signalValue) const {
|
2020-02-06 00:56:35 +00:00
|
|
|
DAWN_TRY(GetDevice()->ValidateIsAlive());
|
2019-02-13 13:09:18 +00:00
|
|
|
DAWN_TRY(GetDevice()->ValidateObject(this));
|
|
|
|
DAWN_TRY(GetDevice()->ValidateObject(fence));
|
2018-12-10 10:03:08 +00:00
|
|
|
|
2019-03-06 22:42:22 +00:00
|
|
|
if (fence->GetQueue() != this) {
|
|
|
|
return DAWN_VALIDATION_ERROR(
|
|
|
|
"Fence must be signaled on the queue on which it was created.");
|
|
|
|
}
|
2018-12-03 16:57:34 +00:00
|
|
|
if (signalValue <= fence->GetSignaledValue()) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Signal value less than or equal to fence signaled value");
|
|
|
|
}
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-06-02 09:24:39 +00:00
|
|
|
MaybeError QueueBase::ValidateCreateFence(const FenceDescriptor* descriptor) const {
|
2020-02-06 00:56:35 +00:00
|
|
|
DAWN_TRY(GetDevice()->ValidateIsAlive());
|
2019-03-06 22:42:22 +00:00
|
|
|
DAWN_TRY(GetDevice()->ValidateObject(this));
|
2020-05-11 20:29:22 +00:00
|
|
|
if (descriptor != nullptr) {
|
|
|
|
DAWN_TRY(ValidateFenceDescriptor(descriptor));
|
|
|
|
}
|
2019-03-06 22:42:22 +00:00
|
|
|
|
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-06-02 09:24:39 +00:00
|
|
|
MaybeError QueueBase::ValidateWriteBuffer(const BufferBase* buffer,
|
|
|
|
uint64_t bufferOffset,
|
|
|
|
size_t size) const {
|
|
|
|
DAWN_TRY(GetDevice()->ValidateIsAlive());
|
|
|
|
DAWN_TRY(GetDevice()->ValidateObject(this));
|
|
|
|
DAWN_TRY(GetDevice()->ValidateObject(buffer));
|
|
|
|
|
|
|
|
if (bufferOffset % 4 != 0) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Queue::WriteBuffer bufferOffset must be a multiple of 4");
|
|
|
|
}
|
|
|
|
if (size % 4 != 0) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Queue::WriteBuffer size must be a multiple of 4");
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t bufferSize = buffer->GetSize();
|
|
|
|
if (bufferOffset > bufferSize || size > (bufferSize - bufferOffset)) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Queue::WriteBuffer out of range");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(buffer->GetUsage() & wgpu::BufferUsage::CopyDst)) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Buffer needs the CopyDst usage bit");
|
|
|
|
}
|
|
|
|
|
2020-08-03 19:22:33 +00:00
|
|
|
DAWN_TRY(buffer->ValidateCanUseOnQueueNow());
|
|
|
|
|
|
|
|
return {};
|
2020-06-02 09:24:39 +00:00
|
|
|
}
|
|
|
|
|
2020-07-08 18:42:30 +00:00
|
|
|
MaybeError QueueBase::ValidateWriteTexture(const TextureCopyView* destination,
|
|
|
|
size_t dataSize,
|
|
|
|
const TextureDataLayout* dataLayout,
|
|
|
|
const Extent3D* writeSize) const {
|
|
|
|
DAWN_TRY(GetDevice()->ValidateIsAlive());
|
|
|
|
DAWN_TRY(GetDevice()->ValidateObject(this));
|
|
|
|
DAWN_TRY(GetDevice()->ValidateObject(destination->texture));
|
|
|
|
|
2020-09-01 18:40:18 +00:00
|
|
|
DAWN_TRY(ValidateTextureCopyView(GetDevice(), *destination, *writeSize));
|
2020-07-08 18:42:30 +00:00
|
|
|
|
|
|
|
if (dataLayout->offset > dataSize) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Queue::WriteTexture out of range");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(destination->texture->GetUsage() & wgpu::TextureUsage::CopyDst)) {
|
|
|
|
return DAWN_VALIDATION_ERROR("Texture needs the CopyDst usage bit");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (destination->texture->GetSampleCount() > 1) {
|
|
|
|
return DAWN_VALIDATION_ERROR("The sample count of textures must be 1");
|
|
|
|
}
|
|
|
|
|
2020-09-02 18:50:09 +00:00
|
|
|
DAWN_TRY(ValidateBufferToTextureCopyRestrictions(*destination));
|
2020-07-17 09:44:46 +00:00
|
|
|
// We validate texture copy range before validating linear texture data,
|
|
|
|
// because in the latter we divide copyExtent.width by blockWidth and
|
|
|
|
// copyExtent.height by blockHeight while the divisibility conditions are
|
|
|
|
// checked in validating texture copy range.
|
|
|
|
DAWN_TRY(ValidateTextureCopyRange(*destination, *writeSize));
|
2020-07-30 15:29:57 +00:00
|
|
|
DAWN_TRY(ValidateLinearTextureData(
|
|
|
|
*dataLayout, dataSize,
|
|
|
|
destination->texture->GetFormat().GetTexelBlockInfo(destination->aspect), *writeSize));
|
2020-07-08 18:42:30 +00:00
|
|
|
|
2020-08-03 19:22:33 +00:00
|
|
|
DAWN_TRY(destination->texture->ValidateCanUseInSubmitNow());
|
|
|
|
|
2020-07-08 18:42:30 +00:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2020-08-13 20:25:39 +00:00
|
|
|
void QueueBase::SubmitInternal(uint32_t commandCount, CommandBufferBase* const* commands) {
|
|
|
|
DeviceBase* device = GetDevice();
|
|
|
|
if (device->ConsumedError(device->ValidateIsAlive())) {
|
|
|
|
// If device is lost, don't let any commands be submitted
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
TRACE_EVENT0(device->GetPlatform(), General, "Queue::Submit");
|
|
|
|
if (device->IsValidationEnabled() &&
|
|
|
|
device->ConsumedError(ValidateSubmit(commandCount, commands))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ASSERT(!IsError());
|
|
|
|
|
|
|
|
if (device->ConsumedError(SubmitImpl(commandCount, commands))) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
device->GetErrorScopeTracker()->TrackUntilLastSubmitComplete(
|
|
|
|
device->GetCurrentErrorScope());
|
|
|
|
}
|
|
|
|
|
2020-07-29 18:37:51 +00:00
|
|
|
void CopyTextureData(uint8_t* dstPointer,
|
|
|
|
const uint8_t* srcPointer,
|
|
|
|
uint32_t depth,
|
|
|
|
uint32_t rowsPerImageInBlock,
|
|
|
|
uint64_t imageAdditionalStride,
|
|
|
|
uint32_t actualBytesPerRow,
|
|
|
|
uint32_t dstBytesPerRow,
|
|
|
|
uint32_t srcBytesPerRow) {
|
|
|
|
bool copyWholeLayer =
|
|
|
|
actualBytesPerRow == dstBytesPerRow && dstBytesPerRow == srcBytesPerRow;
|
|
|
|
bool copyWholeData = copyWholeLayer && imageAdditionalStride == 0;
|
|
|
|
|
|
|
|
if (!copyWholeLayer) { // copy row by row
|
|
|
|
for (uint32_t d = 0; d < depth; ++d) {
|
|
|
|
for (uint32_t h = 0; h < rowsPerImageInBlock; ++h) {
|
|
|
|
memcpy(dstPointer, srcPointer, actualBytesPerRow);
|
|
|
|
dstPointer += dstBytesPerRow;
|
|
|
|
srcPointer += srcBytesPerRow;
|
|
|
|
}
|
|
|
|
srcPointer += imageAdditionalStride;
|
|
|
|
}
|
|
|
|
} else {
|
2020-08-11 12:04:52 +00:00
|
|
|
uint64_t layerSize = uint64_t(rowsPerImageInBlock) * actualBytesPerRow;
|
2020-07-29 18:37:51 +00:00
|
|
|
if (!copyWholeData) { // copy layer by layer
|
|
|
|
for (uint32_t d = 0; d < depth; ++d) {
|
|
|
|
memcpy(dstPointer, srcPointer, layerSize);
|
|
|
|
dstPointer += layerSize;
|
|
|
|
srcPointer += layerSize + imageAdditionalStride;
|
|
|
|
}
|
|
|
|
} else { // do a single copy
|
|
|
|
memcpy(dstPointer, srcPointer, layerSize * depth);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-07-24 14:45:45 +00:00
|
|
|
} // namespace dawn_native
|