Wire: Implement MapWriteAsync
The flow of commands is a bit more involved than for MapReadAsync and goes like this: - C->S MapAsync isWrite = true - S: Call MapWriteAsync - S: MapWriteAsync callback fired - S->C: MapWriteAsyncCallback (no data compared to the read case) - C: Call the MapWriteAsync callback with a zeroed out buffer - C: Application calls unmap. - C->S: UpdateMappedData with the content of the mapped pointer - S: Copy the data in the mapped pointer - C->S: Regular unmap command - S: Call unmap Makes nxt_end2end_tests -w pass all tests. Also duplicates the MapRead wire tests for the write cases
This commit is contained in:
parent
88fb8fa353
commit
5ab96e0d40
|
@ -89,25 +89,33 @@ namespace nxt { namespace wire {
|
||||||
}
|
}
|
||||||
|
|
||||||
void ClearMapRequests(nxtBufferMapAsyncStatus status) {
|
void ClearMapRequests(nxtBufferMapAsyncStatus status) {
|
||||||
for (auto& it : readRequests) {
|
for (auto& it : requests) {
|
||||||
it.second.callback(status, nullptr, it.second.userdata);
|
if (it.second.isWrite) {
|
||||||
|
it.second.writeCallback(status, nullptr, it.second.userdata);
|
||||||
|
} else {
|
||||||
|
it.second.readCallback(status, nullptr, it.second.userdata);
|
||||||
}
|
}
|
||||||
readRequests.clear();
|
}
|
||||||
|
requests.clear();
|
||||||
}
|
}
|
||||||
|
|
||||||
//* We want to defer all the validation to the server, which means we could have multiple
|
//* We want to defer all the validation to the server, which means we could have multiple
|
||||||
//* map request in flight at a single time and need to track them separately.
|
//* map request in flight at a single time and need to track them separately.
|
||||||
//* On well-behaved applications, only one request should exist at a single time.
|
//* On well-behaved applications, only one request should exist at a single time.
|
||||||
struct MapReadRequestData {
|
struct MapRequestData {
|
||||||
nxtBufferMapReadCallback callback = nullptr;
|
nxtBufferMapReadCallback readCallback = nullptr;
|
||||||
|
nxtBufferMapWriteCallback writeCallback = nullptr;
|
||||||
nxtCallbackUserdata userdata = 0;
|
nxtCallbackUserdata userdata = 0;
|
||||||
uint32_t size = 0;
|
uint32_t size = 0;
|
||||||
|
bool isWrite = false;
|
||||||
};
|
};
|
||||||
std::map<uint32_t, MapReadRequestData> readRequests;
|
std::map<uint32_t, MapRequestData> requests;
|
||||||
uint32_t readRequestSerial = 0;
|
uint32_t requestSerial = 0;
|
||||||
|
|
||||||
//* Only one mapped pointer can be active at a time because Unmap clears all the in-flight requests.
|
//* Only one mapped pointer can be active at a time because Unmap clears all the in-flight requests.
|
||||||
void* mappedData = nullptr;
|
void* mappedData = nullptr;
|
||||||
|
size_t mappedDataSize = 0;
|
||||||
|
bool isWriteMapped = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
//* TODO(cwallez@chromium.org): Do something with objects before they are destroyed ?
|
//* TODO(cwallez@chromium.org): Do something with objects before they are destroyed ?
|
||||||
|
@ -314,28 +322,47 @@ namespace nxt { namespace wire {
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
void ClientBufferMapReadAsync(Buffer* buffer, uint32_t start, uint32_t size, nxtBufferMapReadCallback callback, nxtCallbackUserdata userdata) {
|
void ClientBufferMapReadAsync(Buffer* buffer, uint32_t start, uint32_t size, nxtBufferMapReadCallback callback, nxtCallbackUserdata userdata) {
|
||||||
uint32_t serial = buffer->readRequestSerial++;
|
uint32_t serial = buffer->requestSerial++;
|
||||||
ASSERT(buffer->readRequests.find(serial) == buffer->readRequests.end());
|
ASSERT(buffer->requests.find(serial) == buffer->requests.end());
|
||||||
|
|
||||||
Buffer::MapReadRequestData request;
|
Buffer::MapRequestData request;
|
||||||
request.callback = callback;
|
request.readCallback = callback;
|
||||||
request.userdata = userdata;
|
request.userdata = userdata;
|
||||||
request.size = size;
|
request.size = size;
|
||||||
buffer->readRequests[serial] = request;
|
request.isWrite = false;
|
||||||
|
buffer->requests[serial] = request;
|
||||||
|
|
||||||
wire::BufferMapReadAsyncCmd cmd;
|
wire::BufferMapAsyncCmd cmd;
|
||||||
cmd.bufferId = buffer->id;
|
cmd.bufferId = buffer->id;
|
||||||
cmd.requestSerial = serial;
|
cmd.requestSerial = serial;
|
||||||
cmd.start = start;
|
cmd.start = start;
|
||||||
cmd.size = size;
|
cmd.size = size;
|
||||||
|
cmd.isWrite = false;
|
||||||
|
|
||||||
auto allocCmd = static_cast<decltype(cmd)*>(buffer->device->GetCmdSpace(sizeof(cmd)));
|
auto allocCmd = static_cast<decltype(cmd)*>(buffer->device->GetCmdSpace(sizeof(cmd)));
|
||||||
*allocCmd = cmd;
|
*allocCmd = cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ClientBufferMapWriteAsync(Buffer*, uint32_t, uint32_t, nxtBufferMapWriteCallback, nxtCallbackUserdata) {
|
void ClientBufferMapWriteAsync(Buffer* buffer, uint32_t start, uint32_t size, nxtBufferMapWriteCallback callback, nxtCallbackUserdata userdata) {
|
||||||
// TODO(cwallez@chromium.org): Implement the wire for BufferMapWriteAsync
|
uint32_t serial = buffer->requestSerial++;
|
||||||
ASSERT(false);
|
ASSERT(buffer->requests.find(serial) == buffer->requests.end());
|
||||||
|
|
||||||
|
Buffer::MapRequestData request;
|
||||||
|
request.writeCallback = callback;
|
||||||
|
request.userdata = userdata;
|
||||||
|
request.size = size;
|
||||||
|
request.isWrite = true;
|
||||||
|
buffer->requests[serial] = request;
|
||||||
|
|
||||||
|
wire::BufferMapAsyncCmd cmd;
|
||||||
|
cmd.bufferId = buffer->id;
|
||||||
|
cmd.requestSerial = serial;
|
||||||
|
cmd.start = start;
|
||||||
|
cmd.size = size;
|
||||||
|
cmd.isWrite = true;
|
||||||
|
|
||||||
|
auto allocCmd = static_cast<decltype(cmd)*>(buffer->device->GetCmdSpace(sizeof(cmd)));
|
||||||
|
*allocCmd = cmd;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ProxyClientBufferUnmap(nxtBuffer cBuffer) {
|
void ProxyClientBufferUnmap(nxtBuffer cBuffer) {
|
||||||
|
@ -349,6 +376,20 @@ namespace nxt { namespace wire {
|
||||||
//* - Unmap locally on the client
|
//* - Unmap locally on the client
|
||||||
//* - Server -> Client: Result of MapRequest2
|
//* - Server -> Client: Result of MapRequest2
|
||||||
if (buffer->mappedData) {
|
if (buffer->mappedData) {
|
||||||
|
|
||||||
|
// If the buffer was mapped for writing, send the update to the data to the server
|
||||||
|
if (buffer->isWriteMapped) {
|
||||||
|
wire::BufferUpdateMappedDataCmd cmd;
|
||||||
|
cmd.bufferId = buffer->id;
|
||||||
|
cmd.dataLength = static_cast<uint32_t>(buffer->mappedDataSize);
|
||||||
|
|
||||||
|
auto allocCmd = static_cast<decltype(cmd)*>(buffer->device->GetCmdSpace(sizeof(cmd)));
|
||||||
|
*allocCmd = cmd;
|
||||||
|
|
||||||
|
void* dataAlloc = buffer->device->GetCmdSpace(cmd.dataLength);
|
||||||
|
memcpy(dataAlloc, buffer->mappedData, cmd.dataLength);
|
||||||
|
}
|
||||||
|
|
||||||
free(buffer->mappedData);
|
free(buffer->mappedData);
|
||||||
buffer->mappedData = nullptr;
|
buffer->mappedData = nullptr;
|
||||||
}
|
}
|
||||||
|
@ -369,7 +410,7 @@ namespace nxt { namespace wire {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Some commands don't have a custom wire format, but need to be handled manually to update
|
// Some commands don't have a custom wire format, but need to be handled manually to update
|
||||||
// some client-side state tracking. For these we have to functions:
|
// some client-side state tracking. For these we have two functions:
|
||||||
// - An autogenerated Client{{suffix}} method that sends the command on the wire
|
// - An autogenerated Client{{suffix}} method that sends the command on the wire
|
||||||
// - A manual ProxyClient{{suffix}} method that will be inserted in the proctable instead of
|
// - A manual ProxyClient{{suffix}} method that will be inserted in the proctable instead of
|
||||||
// the autogenerated one, and that will have to call Client{{suffix}}
|
// the autogenerated one, and that will have to call Client{{suffix}}
|
||||||
|
@ -412,6 +453,9 @@ namespace nxt { namespace wire {
|
||||||
case ReturnWireCmd::BufferMapReadAsyncCallback:
|
case ReturnWireCmd::BufferMapReadAsyncCallback:
|
||||||
success = HandleBufferMapReadAsyncCallback(&commands, &size);
|
success = HandleBufferMapReadAsyncCallback(&commands, &size);
|
||||||
break;
|
break;
|
||||||
|
case ReturnWireCmd::BufferMapWriteAsyncCallback:
|
||||||
|
success = HandleBufferMapWriteAsyncCallback(&commands, &size);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
success = false;
|
success = false;
|
||||||
}
|
}
|
||||||
|
@ -517,18 +561,22 @@ namespace nxt { namespace wire {
|
||||||
}
|
}
|
||||||
|
|
||||||
//* The requests can have been deleted via an Unmap so this isn't an error.
|
//* The requests can have been deleted via an Unmap so this isn't an error.
|
||||||
auto requestIt = buffer->readRequests.find(cmd->requestSerial);
|
auto requestIt = buffer->requests.find(cmd->requestSerial);
|
||||||
if (requestIt == buffer->readRequests.end()) {
|
if (requestIt == buffer->requests.end()) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//* It is an error for the server to call the read callback when we asked for a map write
|
||||||
|
if (requestIt->second.isWrite) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
auto request = requestIt->second;
|
auto request = requestIt->second;
|
||||||
// Delete the request before calling the callback otherwise the callback could be fired a second time if for example buffer.Unmap() is called inside the callback.
|
//* Delete the request before calling the callback otherwise the callback could be fired a second time. If, for example, buffer.Unmap() is called inside the callback.
|
||||||
buffer->readRequests.erase(requestIt);
|
buffer->requests.erase(requestIt);
|
||||||
|
|
||||||
//* On success, we copy the data locally because the IPC buffer isn't valid outside of this function
|
//* On success, we copy the data locally because the IPC buffer isn't valid outside of this function
|
||||||
if (cmd->status == NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS) {
|
if (cmd->status == NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS) {
|
||||||
|
|
||||||
//* The server didn't send the right amount of data, this is an error and could cause
|
//* The server didn't send the right amount of data, this is an error and could cause
|
||||||
//* the application to crash if we did call the callback.
|
//* the application to crash if we did call the callback.
|
||||||
if (request.size != cmd->dataLength) {
|
if (request.size != cmd->dataLength) {
|
||||||
|
@ -544,12 +592,62 @@ namespace nxt { namespace wire {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
buffer->isWriteMapped = false;
|
||||||
|
buffer->mappedDataSize = request.size;
|
||||||
buffer->mappedData = malloc(request.size);
|
buffer->mappedData = malloc(request.size);
|
||||||
memcpy(buffer->mappedData, requestData, request.size);
|
memcpy(buffer->mappedData, requestData, request.size);
|
||||||
|
|
||||||
request.callback(static_cast<nxtBufferMapAsyncStatus>(cmd->status), buffer->mappedData, request.userdata);
|
request.readCallback(static_cast<nxtBufferMapAsyncStatus>(cmd->status), buffer->mappedData, request.userdata);
|
||||||
} else {
|
} else {
|
||||||
request.callback(static_cast<nxtBufferMapAsyncStatus>(cmd->status), nullptr, request.userdata);
|
request.readCallback(static_cast<nxtBufferMapAsyncStatus>(cmd->status), nullptr, request.userdata);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool HandleBufferMapWriteAsyncCallback(const char** commands, size_t* size) {
|
||||||
|
const auto* cmd = GetCommand<ReturnBufferMapWriteAsyncCallbackCmd>(commands, size);
|
||||||
|
if (cmd == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto* buffer = mDevice->buffer.GetObject(cmd->bufferId);
|
||||||
|
uint32_t bufferSerial = mDevice->buffer.GetSerial(cmd->bufferId);
|
||||||
|
|
||||||
|
//* The buffer might have been deleted or recreated so this isn't an error.
|
||||||
|
if (buffer == nullptr || bufferSerial != cmd->bufferSerial) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
//* The requests can have been deleted via an Unmap so this isn't an error.
|
||||||
|
auto requestIt = buffer->requests.find(cmd->requestSerial);
|
||||||
|
if (requestIt == buffer->requests.end()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
//* It is an error for the server to call the write callback when we asked for a map read
|
||||||
|
if (!requestIt->second.isWrite) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto request = requestIt->second;
|
||||||
|
//* Delete the request before calling the callback otherwise the callback could be fired a second time. If, for example, buffer.Unmap() is called inside the callback.
|
||||||
|
buffer->requests.erase(requestIt);
|
||||||
|
|
||||||
|
//* On success, we copy the data locally because the IPC buffer isn't valid outside of this function
|
||||||
|
if (cmd->status == NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS) {
|
||||||
|
if (buffer->mappedData != nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
buffer->isWriteMapped = true;
|
||||||
|
buffer->mappedDataSize = request.size;
|
||||||
|
buffer->mappedData = malloc(request.size);
|
||||||
|
memset(buffer->mappedData, 0, request.size);
|
||||||
|
|
||||||
|
request.writeCallback(static_cast<nxtBufferMapAsyncStatus>(cmd->status), buffer->mappedData, request.userdata);
|
||||||
|
} else {
|
||||||
|
request.writeCallback(static_cast<nxtBufferMapAsyncStatus>(cmd->status), nullptr, request.userdata);
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -59,7 +59,8 @@ namespace nxt { namespace wire {
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
{{as_MethodSuffix(type.name, Name("destroy"))}},
|
{{as_MethodSuffix(type.name, Name("destroy"))}},
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
BufferMapReadAsync,
|
BufferMapAsync,
|
||||||
|
BufferUpdateMappedDataCmd,
|
||||||
};
|
};
|
||||||
|
|
||||||
{% for type in by_category["object"] %}
|
{% for type in by_category["object"] %}
|
||||||
|
@ -124,6 +125,7 @@ namespace nxt { namespace wire {
|
||||||
{{type.name.CamelCase()}}ErrorCallback,
|
{{type.name.CamelCase()}}ErrorCallback,
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
BufferMapReadAsyncCallback,
|
BufferMapReadAsyncCallback,
|
||||||
|
BufferMapWriteAsyncCallback,
|
||||||
};
|
};
|
||||||
|
|
||||||
//* Command for the server calling a builder status callback.
|
//* Command for the server calling a builder status callback.
|
||||||
|
|
|
@ -27,12 +27,13 @@ namespace nxt { namespace wire {
|
||||||
namespace server {
|
namespace server {
|
||||||
class Server;
|
class Server;
|
||||||
|
|
||||||
struct MapReadUserdata {
|
struct MapUserdata {
|
||||||
Server* server;
|
Server* server;
|
||||||
uint32_t bufferId;
|
uint32_t bufferId;
|
||||||
uint32_t bufferSerial;
|
uint32_t bufferSerial;
|
||||||
uint32_t requestSerial;
|
uint32_t requestSerial;
|
||||||
uint32_t size;
|
uint32_t size;
|
||||||
|
bool isWrite;
|
||||||
};
|
};
|
||||||
|
|
||||||
//* Stores what the backend knows about the type.
|
//* Stores what the backend knows about the type.
|
||||||
|
@ -54,6 +55,10 @@ namespace nxt { namespace wire {
|
||||||
//* Whether this object has been allocated, used by the KnownObjects queries
|
//* Whether this object has been allocated, used by the KnownObjects queries
|
||||||
//* TODO(cwallez@chromium.org): make this an internal bit vector in KnownObjects.
|
//* TODO(cwallez@chromium.org): make this an internal bit vector in KnownObjects.
|
||||||
bool allocated;
|
bool allocated;
|
||||||
|
|
||||||
|
//* TODO(cwallez@chromium.org): this is only useful for buffers
|
||||||
|
void* mappedData = nullptr;
|
||||||
|
size_t mappedDataSize = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
//* Keeps track of the mapping between client IDs and backend objects.
|
//* Keeps track of the mapping between client IDs and backend objects.
|
||||||
|
@ -143,6 +148,7 @@ namespace nxt { namespace wire {
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
void ForwardBufferMapReadAsync(nxtBufferMapAsyncStatus status, const void* ptr, nxtCallbackUserdata userdata);
|
void ForwardBufferMapReadAsync(nxtBufferMapAsyncStatus status, const void* ptr, nxtCallbackUserdata userdata);
|
||||||
|
void ForwardBufferMapWriteAsync(nxtBufferMapAsyncStatus status, void* ptr, nxtCallbackUserdata userdata);
|
||||||
|
|
||||||
// A really really simple implementation of the DeserializeAllocator. It's main feature
|
// A really really simple implementation of the DeserializeAllocator. It's main feature
|
||||||
// is that it has some inline storage so as to avoid allocations for the majority of
|
// is that it has some inline storage so as to avoid allocations for the majority of
|
||||||
|
@ -253,7 +259,7 @@ namespace nxt { namespace wire {
|
||||||
}
|
}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
void OnMapReadAsyncCallback(nxtBufferMapAsyncStatus status, const void* ptr, MapReadUserdata* data) {
|
void OnMapReadAsyncCallback(nxtBufferMapAsyncStatus status, const void* ptr, MapUserdata* data) {
|
||||||
ReturnBufferMapReadAsyncCallbackCmd cmd;
|
ReturnBufferMapReadAsyncCallbackCmd cmd;
|
||||||
cmd.bufferId = data->bufferId;
|
cmd.bufferId = data->bufferId;
|
||||||
cmd.bufferSerial = data->bufferSerial;
|
cmd.bufferSerial = data->bufferSerial;
|
||||||
|
@ -274,6 +280,27 @@ namespace nxt { namespace wire {
|
||||||
delete data;
|
delete data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void OnMapWriteAsyncCallback(nxtBufferMapAsyncStatus status, void* ptr, MapUserdata* data) {
|
||||||
|
ReturnBufferMapWriteAsyncCallbackCmd cmd;
|
||||||
|
cmd.bufferId = data->bufferId;
|
||||||
|
cmd.bufferSerial = data->bufferSerial;
|
||||||
|
cmd.requestSerial = data->requestSerial;
|
||||||
|
cmd.status = status;
|
||||||
|
|
||||||
|
auto allocCmd = static_cast<ReturnBufferMapWriteAsyncCallbackCmd*>(GetCmdSpace(sizeof(cmd)));
|
||||||
|
*allocCmd = cmd;
|
||||||
|
|
||||||
|
if (status == NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS) {
|
||||||
|
auto* selfData = mKnownBuffer.Get(data->bufferId);
|
||||||
|
ASSERT(selfData != nullptr);
|
||||||
|
|
||||||
|
selfData->mappedData = ptr;
|
||||||
|
selfData->mappedDataSize = data->size;
|
||||||
|
}
|
||||||
|
|
||||||
|
delete data;
|
||||||
|
}
|
||||||
|
|
||||||
const char* HandleCommands(const char* commands, size_t size) override {
|
const char* HandleCommands(const char* commands, size_t size) override {
|
||||||
mProcs.deviceTick(mKnownDevice.Get(1)->handle);
|
mProcs.deviceTick(mKnownDevice.Get(1)->handle);
|
||||||
|
|
||||||
|
@ -294,8 +321,11 @@ namespace nxt { namespace wire {
|
||||||
success = Handle{{Suffix}}(&commands, &size);
|
success = Handle{{Suffix}}(&commands, &size);
|
||||||
break;
|
break;
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
case WireCmd::BufferMapReadAsync:
|
case WireCmd::BufferMapAsync:
|
||||||
success = HandleBufferMapReadAsync(&commands, &size);
|
success = HandleBufferMapAsync(&commands, &size);
|
||||||
|
break;
|
||||||
|
case WireCmd::BufferUpdateMappedDataCmd:
|
||||||
|
success = HandleBufferUpdateMappedData(&commands, &size);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
|
@ -351,17 +381,34 @@ namespace nxt { namespace wire {
|
||||||
//* Checks there is enough data left, updates the buffer / size and returns
|
//* Checks there is enough data left, updates the buffer / size and returns
|
||||||
//* the command (or nullptr for an error).
|
//* the command (or nullptr for an error).
|
||||||
template <typename T>
|
template <typename T>
|
||||||
static const T* GetCommand(const char** commands, size_t* size) {
|
static const T* GetData(const char** buffer, size_t* size, size_t count) {
|
||||||
if (*size < sizeof(T)) {
|
// TODO(cwallez@chromium.org): Check for overflow
|
||||||
|
size_t totalSize = count * sizeof(T);
|
||||||
|
if (*size < totalSize) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
const T* cmd = reinterpret_cast<const T*>(*commands);
|
const T* data = reinterpret_cast<const T*>(*buffer);
|
||||||
|
|
||||||
*commands += sizeof(T);
|
*buffer += totalSize;
|
||||||
*size -= sizeof(T);
|
*size -= totalSize;
|
||||||
|
|
||||||
return cmd;
|
return data;
|
||||||
|
}
|
||||||
|
template <typename T>
|
||||||
|
static const T* GetCommand(const char** commands, size_t* size) {
|
||||||
|
return GetData<T>(commands, size, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
{% set custom_pre_handler_commands = ["BufferUnmap"] %}
|
||||||
|
|
||||||
|
bool PreHandleBufferUnmap(const BufferUnmapCmd& cmd) {
|
||||||
|
auto* selfData = mKnownBuffer.Get(cmd.selfId);
|
||||||
|
ASSERT(selfData != nullptr);
|
||||||
|
|
||||||
|
selfData->mappedData = nullptr;
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
//* Implementation of the command handlers
|
//* Implementation of the command handlers
|
||||||
|
@ -379,6 +426,12 @@ namespace nxt { namespace wire {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{% if Suffix in custom_pre_handler_commands %}
|
||||||
|
if (!PreHandle{{Suffix}}(cmd)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
//* Unpack 'self'
|
//* Unpack 'self'
|
||||||
auto* selfData = mKnown{{type.name.CamelCase()}}.Get(cmd.selfId);
|
auto* selfData = mKnown{{type.name.CamelCase()}}.Get(cmd.selfId);
|
||||||
ASSERT(selfData != nullptr);
|
ASSERT(selfData != nullptr);
|
||||||
|
@ -470,10 +523,10 @@ namespace nxt { namespace wire {
|
||||||
}
|
}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
bool HandleBufferMapReadAsync(const char** commands, size_t* size) {
|
bool HandleBufferMapAsync(const char** commands, size_t* size) {
|
||||||
//* These requests are just forwarded to the buffer, with userdata containing what the client
|
//* These requests are just forwarded to the buffer, with userdata containing what the client
|
||||||
//* will require in the return command.
|
//* will require in the return command.
|
||||||
const auto* cmd = GetCommand<BufferMapReadAsyncCmd>(commands, size);
|
const auto* cmd = GetCommand<BufferMapAsyncCmd>(commands, size);
|
||||||
if (cmd == nullptr) {
|
if (cmd == nullptr) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -482,28 +535,63 @@ namespace nxt { namespace wire {
|
||||||
uint32_t requestSerial = cmd->requestSerial;
|
uint32_t requestSerial = cmd->requestSerial;
|
||||||
uint32_t requestSize = cmd->size;
|
uint32_t requestSize = cmd->size;
|
||||||
uint32_t requestStart = cmd->start;
|
uint32_t requestStart = cmd->start;
|
||||||
|
bool isWrite = cmd->isWrite;
|
||||||
|
|
||||||
auto* buffer = mKnownBuffer.Get(bufferId);
|
auto* buffer = mKnownBuffer.Get(bufferId);
|
||||||
if (buffer == nullptr) {
|
if (buffer == nullptr) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto* data = new MapReadUserdata;
|
auto* data = new MapUserdata;
|
||||||
data->server = this;
|
data->server = this;
|
||||||
data->bufferId = bufferId;
|
data->bufferId = bufferId;
|
||||||
data->bufferSerial = buffer->serial;
|
data->bufferSerial = buffer->serial;
|
||||||
data->requestSerial = requestSerial;
|
data->requestSerial = requestSerial;
|
||||||
data->size = requestSize;
|
data->size = requestSize;
|
||||||
|
data->isWrite = isWrite;
|
||||||
|
|
||||||
auto userdata = static_cast<uint64_t>(reinterpret_cast<uintptr_t>(data));
|
auto userdata = static_cast<uint64_t>(reinterpret_cast<uintptr_t>(data));
|
||||||
|
|
||||||
if (!buffer->valid) {
|
if (!buffer->valid) {
|
||||||
//* Fake the buffer returning a failure, data will be freed in this call.
|
//* Fake the buffer returning a failure, data will be freed in this call.
|
||||||
|
if (isWrite) {
|
||||||
|
ForwardBufferMapWriteAsync(NXT_BUFFER_MAP_ASYNC_STATUS_ERROR, nullptr, userdata);
|
||||||
|
} else {
|
||||||
ForwardBufferMapReadAsync(NXT_BUFFER_MAP_ASYNC_STATUS_ERROR, nullptr, userdata);
|
ForwardBufferMapReadAsync(NXT_BUFFER_MAP_ASYNC_STATUS_ERROR, nullptr, userdata);
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (isWrite) {
|
||||||
|
mProcs.bufferMapWriteAsync(buffer->handle, requestStart, requestSize, ForwardBufferMapWriteAsync, userdata);
|
||||||
|
} else {
|
||||||
mProcs.bufferMapReadAsync(buffer->handle, requestStart, requestSize, ForwardBufferMapReadAsync, userdata);
|
mProcs.bufferMapReadAsync(buffer->handle, requestStart, requestSize, ForwardBufferMapReadAsync, userdata);
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool HandleBufferUpdateMappedData(const char** commands, size_t* size) {
|
||||||
|
const auto* cmd = GetCommand<BufferUpdateMappedDataCmd>(commands, size);
|
||||||
|
if (cmd == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
ObjectId bufferId = cmd->bufferId;
|
||||||
|
size_t dataLength = cmd->dataLength;
|
||||||
|
|
||||||
|
auto* buffer = mKnownBuffer.Get(bufferId);
|
||||||
|
if (buffer == nullptr || !buffer->valid || buffer->mappedData == nullptr ||
|
||||||
|
buffer->mappedDataSize != dataLength) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char* data = GetData<char>(commands, size, dataLength);
|
||||||
|
if (data == nullptr) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
memcpy(buffer->mappedData, data, dataLength);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -524,9 +612,14 @@ namespace nxt { namespace wire {
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
void ForwardBufferMapReadAsync(nxtBufferMapAsyncStatus status, const void* ptr, nxtCallbackUserdata userdata) {
|
void ForwardBufferMapReadAsync(nxtBufferMapAsyncStatus status, const void* ptr, nxtCallbackUserdata userdata) {
|
||||||
auto data = reinterpret_cast<MapReadUserdata*>(static_cast<uintptr_t>(userdata));
|
auto data = reinterpret_cast<MapUserdata*>(static_cast<uintptr_t>(userdata));
|
||||||
data->server->OnMapReadAsyncCallback(status, ptr, data);
|
data->server->OnMapReadAsyncCallback(status, ptr, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ForwardBufferMapWriteAsync(nxtBufferMapAsyncStatus status, void* ptr, nxtCallbackUserdata userdata) {
|
||||||
|
auto data = reinterpret_cast<MapUserdata*>(static_cast<uintptr_t>(userdata));
|
||||||
|
data->server->OnMapWriteAsyncCallback(status, ptr, data);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
CommandHandler* NewServerCommandHandler(nxtDevice device, const nxtProcTable& procs, CommandSerializer* serializer) {
|
CommandHandler* NewServerCommandHandler(nxtDevice device, const nxtProcTable& procs, CommandSerializer* serializer) {
|
||||||
|
|
|
@ -97,7 +97,20 @@ class MockBufferMapReadCallback {
|
||||||
static MockBufferMapReadCallback* mockBufferMapReadCallback = nullptr;
|
static MockBufferMapReadCallback* mockBufferMapReadCallback = nullptr;
|
||||||
static void ToMockBufferMapReadCallback(nxtBufferMapAsyncStatus status, const void* ptr, nxtCallbackUserdata userdata) {
|
static void ToMockBufferMapReadCallback(nxtBufferMapAsyncStatus status, const void* ptr, nxtCallbackUserdata userdata) {
|
||||||
// Assume the data is uint32_t to make writing matchers easier
|
// Assume the data is uint32_t to make writing matchers easier
|
||||||
mockBufferMapReadCallback->Call(status, reinterpret_cast<const uint32_t*>(ptr), userdata);
|
mockBufferMapReadCallback->Call(status, static_cast<const uint32_t*>(ptr), userdata);
|
||||||
|
}
|
||||||
|
|
||||||
|
class MockBufferMapWriteCallback {
|
||||||
|
public:
|
||||||
|
MOCK_METHOD3(Call, void(nxtBufferMapAsyncStatus status, uint32_t* ptr, nxtCallbackUserdata userdata));
|
||||||
|
};
|
||||||
|
|
||||||
|
static MockBufferMapWriteCallback* mockBufferMapWriteCallback = nullptr;
|
||||||
|
uint32_t* lastMapWritePointer = nullptr;
|
||||||
|
static void ToMockBufferMapWriteCallback(nxtBufferMapAsyncStatus status, void* ptr, nxtCallbackUserdata userdata) {
|
||||||
|
// Assume the data is uint32_t to make writing matchers easier
|
||||||
|
lastMapWritePointer = static_cast<uint32_t*>(ptr);
|
||||||
|
mockBufferMapWriteCallback->Call(status, lastMapWritePointer, userdata);
|
||||||
}
|
}
|
||||||
|
|
||||||
class WireTestsBase : public Test {
|
class WireTestsBase : public Test {
|
||||||
|
@ -110,6 +123,7 @@ class WireTestsBase : public Test {
|
||||||
mockDeviceErrorCallback = new MockDeviceErrorCallback;
|
mockDeviceErrorCallback = new MockDeviceErrorCallback;
|
||||||
mockBuilderErrorCallback = new MockBuilderErrorCallback;
|
mockBuilderErrorCallback = new MockBuilderErrorCallback;
|
||||||
mockBufferMapReadCallback = new MockBufferMapReadCallback;
|
mockBufferMapReadCallback = new MockBufferMapReadCallback;
|
||||||
|
mockBufferMapWriteCallback = new MockBufferMapWriteCallback;
|
||||||
|
|
||||||
nxtProcTable mockProcs;
|
nxtProcTable mockProcs;
|
||||||
nxtDevice mockDevice;
|
nxtDevice mockDevice;
|
||||||
|
@ -145,6 +159,7 @@ class WireTestsBase : public Test {
|
||||||
delete mockDeviceErrorCallback;
|
delete mockDeviceErrorCallback;
|
||||||
delete mockBuilderErrorCallback;
|
delete mockBuilderErrorCallback;
|
||||||
delete mockBufferMapReadCallback;
|
delete mockBufferMapReadCallback;
|
||||||
|
delete mockBufferMapWriteCallback;
|
||||||
}
|
}
|
||||||
|
|
||||||
void FlushClient() {
|
void FlushClient() {
|
||||||
|
@ -658,8 +673,10 @@ class WireBufferMappingTests : public WireTestsBase {
|
||||||
nxtBuffer errorBuffer;
|
nxtBuffer errorBuffer;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Check mapping a succesfully created buffer
|
// MapRead-specific tests
|
||||||
TEST_F(WireBufferMappingTests, MappingSuccessBuffer) {
|
|
||||||
|
// Check mapping for reading a succesfully created buffer
|
||||||
|
TEST_F(WireBufferMappingTests, MappingForReadSuccessBuffer) {
|
||||||
nxtCallbackUserdata userdata = 8653;
|
nxtCallbackUserdata userdata = 8653;
|
||||||
nxtBufferMapReadAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
nxtBufferMapReadAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
||||||
|
|
||||||
|
@ -683,8 +700,8 @@ TEST_F(WireBufferMappingTests, MappingSuccessBuffer) {
|
||||||
FlushClient();
|
FlushClient();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check that things work correctly when a validation error happens when mapping the buffer
|
// Check that things work correctly when a validation error happens when mapping the buffer for reading
|
||||||
TEST_F(WireBufferMappingTests, ErrorWhileMapping) {
|
TEST_F(WireBufferMappingTests, ErrorWhileMappingForRead) {
|
||||||
nxtCallbackUserdata userdata = 8654;
|
nxtCallbackUserdata userdata = 8654;
|
||||||
nxtBufferMapReadAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
nxtBufferMapReadAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
||||||
|
|
||||||
|
@ -701,8 +718,8 @@ TEST_F(WireBufferMappingTests, ErrorWhileMapping) {
|
||||||
FlushServer();
|
FlushServer();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check mapping a buffer that didn't get created on the server side
|
// Check mapping for reading a buffer that didn't get created on the server side
|
||||||
TEST_F(WireBufferMappingTests, MappingErrorBuffer) {
|
TEST_F(WireBufferMappingTests, MappingForReadErrorBuffer) {
|
||||||
nxtCallbackUserdata userdata = 8655;
|
nxtCallbackUserdata userdata = 8655;
|
||||||
nxtBufferMapReadAsync(errorBuffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
nxtBufferMapReadAsync(errorBuffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
||||||
|
|
||||||
|
@ -718,8 +735,8 @@ TEST_F(WireBufferMappingTests, MappingErrorBuffer) {
|
||||||
FlushClient();
|
FlushClient();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check that the callback is called with UNKNOWN when the buffer is destroyed before the request is finished
|
// Check that the map read callback is called with UNKNOWN when the buffer is destroyed before the request is finished
|
||||||
TEST_F(WireBufferMappingTests, DestroyBeforeRequestEnd) {
|
TEST_F(WireBufferMappingTests, DestroyBeforeReadRequestEnd) {
|
||||||
nxtCallbackUserdata userdata = 8656;
|
nxtCallbackUserdata userdata = 8656;
|
||||||
nxtBufferMapReadAsync(errorBuffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
nxtBufferMapReadAsync(errorBuffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
||||||
|
|
||||||
|
@ -729,8 +746,8 @@ TEST_F(WireBufferMappingTests, DestroyBeforeRequestEnd) {
|
||||||
nxtBufferRelease(errorBuffer);
|
nxtBufferRelease(errorBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check the callback is called with UNKNOWN when the map request would have worked, but Unmap was called
|
// Check the map read callback is called with UNKNOWN when the map request would have worked, but Unmap was called
|
||||||
TEST_F(WireBufferMappingTests, UnmapCalledTooEarly) {
|
TEST_F(WireBufferMappingTests, UnmapCalledTooEarlyForRead) {
|
||||||
nxtCallbackUserdata userdata = 8657;
|
nxtCallbackUserdata userdata = 8657;
|
||||||
nxtBufferMapReadAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
nxtBufferMapReadAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
||||||
|
|
||||||
|
@ -751,8 +768,8 @@ TEST_F(WireBufferMappingTests, UnmapCalledTooEarly) {
|
||||||
FlushServer();
|
FlushServer();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check that an error callback gets nullptr while a buffer is already mapped
|
// Check that an error map read callback gets nullptr while a buffer is already mapped
|
||||||
TEST_F(WireBufferMappingTests, MappingErrorWhileAlreadyMappedGetsNullptr) {
|
TEST_F(WireBufferMappingTests, MappingForReadingErrorWhileAlreadyMappedGetsNullptr) {
|
||||||
// Successful map
|
// Successful map
|
||||||
nxtCallbackUserdata userdata = 34098;
|
nxtCallbackUserdata userdata = 34098;
|
||||||
nxtBufferMapReadAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
nxtBufferMapReadAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapReadCallback, userdata);
|
||||||
|
@ -838,3 +855,199 @@ TEST_F(WireBufferMappingTests, DestroyInsideMapReadCallback) {
|
||||||
|
|
||||||
FlushClient();
|
FlushClient();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// MapWrite-specific tests
|
||||||
|
|
||||||
|
// Check mapping for writing a succesfully created buffer
|
||||||
|
TEST_F(WireBufferMappingTests, MappingForWriteSuccessBuffer) {
|
||||||
|
nxtCallbackUserdata userdata = 8653;
|
||||||
|
nxtBufferMapWriteAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapWriteCallback, userdata);
|
||||||
|
|
||||||
|
uint32_t serverBufferContent = 31337;
|
||||||
|
uint32_t updatedContent = 4242;
|
||||||
|
uint32_t zero = 0;
|
||||||
|
|
||||||
|
EXPECT_CALL(api, OnBufferMapWriteAsyncCallback(apiBuffer, 40, sizeof(uint32_t), _, _))
|
||||||
|
.WillOnce(InvokeWithoutArgs([&]() {
|
||||||
|
api.CallMapWriteCallback(apiBuffer, NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS, &serverBufferContent);
|
||||||
|
}));
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
|
||||||
|
// The map write callback always gets a buffer full of zeroes.
|
||||||
|
EXPECT_CALL(*mockBufferMapWriteCallback, Call(NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS, Pointee(Eq(zero)), userdata))
|
||||||
|
.Times(1);
|
||||||
|
|
||||||
|
FlushServer();
|
||||||
|
|
||||||
|
// Write something to the mapped pointer
|
||||||
|
*lastMapWritePointer = updatedContent;
|
||||||
|
|
||||||
|
nxtBufferUnmap(buffer);
|
||||||
|
EXPECT_CALL(api, BufferUnmap(apiBuffer))
|
||||||
|
.Times(1);
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
|
||||||
|
// After the buffer is unmapped, the content of the buffer is updated on the server
|
||||||
|
ASSERT_EQ(serverBufferContent, updatedContent);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that things work correctly when a validation error happens when mapping the buffer for writing
|
||||||
|
TEST_F(WireBufferMappingTests, ErrorWhileMappingForWrite) {
|
||||||
|
nxtCallbackUserdata userdata = 8654;
|
||||||
|
nxtBufferMapWriteAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapWriteCallback, userdata);
|
||||||
|
|
||||||
|
EXPECT_CALL(api, OnBufferMapWriteAsyncCallback(apiBuffer, 40, sizeof(uint32_t), _, _))
|
||||||
|
.WillOnce(InvokeWithoutArgs([&]() {
|
||||||
|
api.CallMapWriteCallback(apiBuffer, NXT_BUFFER_MAP_ASYNC_STATUS_ERROR, nullptr);
|
||||||
|
}));
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
|
||||||
|
EXPECT_CALL(*mockBufferMapWriteCallback, Call(NXT_BUFFER_MAP_ASYNC_STATUS_ERROR, nullptr, userdata))
|
||||||
|
.Times(1);
|
||||||
|
|
||||||
|
FlushServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check mapping for writing a buffer that didn't get created on the server side
|
||||||
|
TEST_F(WireBufferMappingTests, MappingForWriteErrorBuffer) {
|
||||||
|
nxtCallbackUserdata userdata = 8655;
|
||||||
|
nxtBufferMapWriteAsync(errorBuffer, 40, sizeof(uint32_t), ToMockBufferMapWriteCallback, userdata);
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
|
||||||
|
EXPECT_CALL(*mockBufferMapWriteCallback, Call(NXT_BUFFER_MAP_ASYNC_STATUS_ERROR, nullptr, userdata))
|
||||||
|
.Times(1);
|
||||||
|
|
||||||
|
FlushServer();
|
||||||
|
|
||||||
|
nxtBufferUnmap(errorBuffer);
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that the map write callback is called with UNKNOWN when the buffer is destroyed before the request is finished
|
||||||
|
TEST_F(WireBufferMappingTests, DestroyBeforeWriteRequestEnd) {
|
||||||
|
nxtCallbackUserdata userdata = 8656;
|
||||||
|
nxtBufferMapWriteAsync(errorBuffer, 40, sizeof(uint32_t), ToMockBufferMapWriteCallback, userdata);
|
||||||
|
|
||||||
|
EXPECT_CALL(*mockBufferMapWriteCallback, Call(NXT_BUFFER_MAP_ASYNC_STATUS_UNKNOWN, nullptr, userdata))
|
||||||
|
.Times(1);
|
||||||
|
|
||||||
|
nxtBufferRelease(errorBuffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check the map read callback is called with UNKNOWN when the map request would have worked, but Unmap was called
|
||||||
|
TEST_F(WireBufferMappingTests, UnmapCalledTooEarlyForWrite) {
|
||||||
|
nxtCallbackUserdata userdata = 8657;
|
||||||
|
nxtBufferMapWriteAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapWriteCallback, userdata);
|
||||||
|
|
||||||
|
uint32_t bufferContent = 31337;
|
||||||
|
EXPECT_CALL(api, OnBufferMapWriteAsyncCallback(apiBuffer, 40, sizeof(uint32_t), _, _))
|
||||||
|
.WillOnce(InvokeWithoutArgs([&]() {
|
||||||
|
api.CallMapWriteCallback(apiBuffer, NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS, &bufferContent);
|
||||||
|
}));
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
|
||||||
|
// Oh no! We are calling Unmap too early!
|
||||||
|
EXPECT_CALL(*mockBufferMapWriteCallback, Call(NXT_BUFFER_MAP_ASYNC_STATUS_UNKNOWN, nullptr, userdata))
|
||||||
|
.Times(1);
|
||||||
|
nxtBufferUnmap(buffer);
|
||||||
|
|
||||||
|
// The callback shouldn't get called, even when the request succeeded on the server side
|
||||||
|
FlushServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check that an error map read callback gets nullptr while a buffer is already mapped
|
||||||
|
TEST_F(WireBufferMappingTests, MappingForWritingErrorWhileAlreadyMappedGetsNullptr) {
|
||||||
|
// Successful map
|
||||||
|
nxtCallbackUserdata userdata = 34098;
|
||||||
|
nxtBufferMapWriteAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapWriteCallback, userdata);
|
||||||
|
|
||||||
|
uint32_t bufferContent = 31337;
|
||||||
|
uint32_t zero = 0;
|
||||||
|
EXPECT_CALL(api, OnBufferMapWriteAsyncCallback(apiBuffer, 40, sizeof(uint32_t), _, _))
|
||||||
|
.WillOnce(InvokeWithoutArgs([&]() {
|
||||||
|
api.CallMapWriteCallback(apiBuffer, NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS, &bufferContent);
|
||||||
|
}))
|
||||||
|
.RetiresOnSaturation();
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
|
||||||
|
EXPECT_CALL(*mockBufferMapWriteCallback, Call(NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS, Pointee(Eq(zero)), userdata))
|
||||||
|
.Times(1);
|
||||||
|
|
||||||
|
FlushServer();
|
||||||
|
|
||||||
|
// Map failure while the buffer is already mapped
|
||||||
|
userdata ++;
|
||||||
|
nxtBufferMapWriteAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapWriteCallback, userdata);
|
||||||
|
EXPECT_CALL(api, OnBufferMapWriteAsyncCallback(apiBuffer, 40, sizeof(uint32_t), _, _))
|
||||||
|
.WillOnce(InvokeWithoutArgs([&]() {
|
||||||
|
api.CallMapWriteCallback(apiBuffer, NXT_BUFFER_MAP_ASYNC_STATUS_ERROR, nullptr);
|
||||||
|
}));
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
|
||||||
|
EXPECT_CALL(*mockBufferMapWriteCallback, Call(NXT_BUFFER_MAP_ASYNC_STATUS_ERROR, nullptr, userdata))
|
||||||
|
.Times(1);
|
||||||
|
|
||||||
|
FlushServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test that the MapWriteCallback isn't fired twice when unmap() is called inside the callback
|
||||||
|
TEST_F(WireBufferMappingTests, UnmapInsideMapWriteCallback) {
|
||||||
|
nxtCallbackUserdata userdata = 2039;
|
||||||
|
nxtBufferMapWriteAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapWriteCallback, userdata);
|
||||||
|
|
||||||
|
uint32_t bufferContent = 31337;
|
||||||
|
uint32_t zero = 0;
|
||||||
|
EXPECT_CALL(api, OnBufferMapWriteAsyncCallback(apiBuffer, 40, sizeof(uint32_t), _, _))
|
||||||
|
.WillOnce(InvokeWithoutArgs([&]() {
|
||||||
|
api.CallMapWriteCallback(apiBuffer, NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS, &bufferContent);
|
||||||
|
}));
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
|
||||||
|
EXPECT_CALL(*mockBufferMapWriteCallback, Call(NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS, Pointee(Eq(zero)), userdata))
|
||||||
|
.WillOnce(InvokeWithoutArgs([&]() {
|
||||||
|
nxtBufferUnmap(buffer);
|
||||||
|
}));
|
||||||
|
|
||||||
|
FlushServer();
|
||||||
|
|
||||||
|
EXPECT_CALL(api, BufferUnmap(apiBuffer))
|
||||||
|
.Times(1);
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test that the MapWriteCallback isn't fired twice the buffer external refcount reaches 0 in the callback
|
||||||
|
TEST_F(WireBufferMappingTests, DestroyInsideMapWriteCallback) {
|
||||||
|
nxtCallbackUserdata userdata = 2039;
|
||||||
|
nxtBufferMapWriteAsync(buffer, 40, sizeof(uint32_t), ToMockBufferMapWriteCallback, userdata);
|
||||||
|
|
||||||
|
uint32_t bufferContent = 31337;
|
||||||
|
uint32_t zero = 0;
|
||||||
|
EXPECT_CALL(api, OnBufferMapWriteAsyncCallback(apiBuffer, 40, sizeof(uint32_t), _, _))
|
||||||
|
.WillOnce(InvokeWithoutArgs([&]() {
|
||||||
|
api.CallMapWriteCallback(apiBuffer, NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS, &bufferContent);
|
||||||
|
}));
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
|
||||||
|
EXPECT_CALL(*mockBufferMapWriteCallback, Call(NXT_BUFFER_MAP_ASYNC_STATUS_SUCCESS, Pointee(Eq(zero)), userdata))
|
||||||
|
.WillOnce(InvokeWithoutArgs([&]() {
|
||||||
|
nxtBufferRelease(buffer);
|
||||||
|
}));
|
||||||
|
|
||||||
|
FlushServer();
|
||||||
|
|
||||||
|
EXPECT_CALL(api, BufferRelease(apiBuffer))
|
||||||
|
.Times(1);
|
||||||
|
|
||||||
|
FlushClient();
|
||||||
|
}
|
||||||
|
|
|
@ -27,25 +27,42 @@ namespace nxt { namespace wire {
|
||||||
size_t messageStrlen;
|
size_t messageStrlen;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct BufferMapReadAsyncCmd {
|
struct BufferMapAsyncCmd {
|
||||||
wire::WireCmd commandId = WireCmd::BufferMapReadAsync;
|
wire::WireCmd commandId = WireCmd::BufferMapAsync;
|
||||||
|
|
||||||
uint32_t bufferId;
|
ObjectId bufferId;
|
||||||
uint32_t requestSerial;
|
ObjectSerial requestSerial;
|
||||||
uint32_t start;
|
uint32_t start;
|
||||||
uint32_t size;
|
uint32_t size;
|
||||||
|
bool isWrite;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ReturnBufferMapReadAsyncCallbackCmd {
|
struct ReturnBufferMapReadAsyncCallbackCmd {
|
||||||
wire::ReturnWireCmd commandId = ReturnWireCmd::BufferMapReadAsyncCallback;
|
wire::ReturnWireCmd commandId = ReturnWireCmd::BufferMapReadAsyncCallback;
|
||||||
|
|
||||||
uint32_t bufferId;
|
ObjectId bufferId;
|
||||||
uint32_t bufferSerial;
|
ObjectSerial bufferSerial;
|
||||||
uint32_t requestSerial;
|
uint32_t requestSerial;
|
||||||
uint32_t status;
|
uint32_t status;
|
||||||
uint32_t dataLength;
|
uint32_t dataLength;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct ReturnBufferMapWriteAsyncCallbackCmd {
|
||||||
|
wire::ReturnWireCmd commandId = ReturnWireCmd::BufferMapWriteAsyncCallback;
|
||||||
|
|
||||||
|
ObjectId bufferId;
|
||||||
|
ObjectSerial bufferSerial;
|
||||||
|
uint32_t requestSerial;
|
||||||
|
uint32_t status;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct BufferUpdateMappedDataCmd {
|
||||||
|
wire::WireCmd commandId = WireCmd::BufferUpdateMappedDataCmd;
|
||||||
|
|
||||||
|
ObjectId bufferId;
|
||||||
|
uint32_t dataLength;
|
||||||
|
};
|
||||||
|
|
||||||
}} // namespace nxt::wire
|
}} // namespace nxt::wire
|
||||||
|
|
||||||
#endif // WIRE_WIRECMD_H_
|
#endif // WIRE_WIRECMD_H_
|
||||||
|
|
Loading…
Reference in New Issue