mirror of
https://gitlab.freedesktop.org/mesa/mesa.git
synced 2026-05-22 08:48:07 +02:00
Mainly to leave main ring thread prio to default. Signed-off-by: Yiwei Zhang <zzyiwei@chromium.org> Part-of: <https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/30279>
138 lines
3.7 KiB
C
138 lines
3.7 KiB
C
/*
|
|
* Copyright 2021 Google LLC
|
|
* SPDX-License-Identifier: MIT
|
|
*/
|
|
|
|
#ifndef VN_RING_H
|
|
#define VN_RING_H
|
|
|
|
#include "vn_common.h"
|
|
|
|
#include "vn_cs.h"
|
|
|
|
/**
|
|
* A ring is a single-producer and single-consumer circular buffer. The data
|
|
* in the buffer are produced and consumed in order. An externally-defined
|
|
* mechanism is required for ring setup and notifications in both directions.
|
|
* Notifications for new data from the producer are needed only when the
|
|
* consumer is not actively polling, which is indicated by the ring status.
|
|
*
|
|
* For venus, the data are plain venus commands. When a venus command is
|
|
* consumed from the ring's perspective, there can still be ongoing CPU and/or
|
|
* GPU works. This is not an issue when the works generated by following
|
|
* venus commands are correctly queued after the ongoing works. There are
|
|
* also venus commands that facilitate polling or waiting for ongoing works.
|
|
*/
|
|
|
|
/* the layout of a ring in a shmem */
|
|
struct vn_ring_layout {
|
|
size_t head_offset;
|
|
size_t tail_offset;
|
|
size_t status_offset;
|
|
|
|
size_t buffer_offset;
|
|
size_t buffer_size;
|
|
|
|
size_t extra_offset;
|
|
size_t extra_size;
|
|
|
|
size_t shmem_size;
|
|
};
|
|
|
|
void
|
|
vn_ring_get_layout(size_t buf_size,
|
|
size_t extra_size,
|
|
struct vn_ring_layout *layout);
|
|
|
|
struct vn_ring *
|
|
vn_ring_create(struct vn_instance *instance,
|
|
const struct vn_ring_layout *layout,
|
|
uint8_t direct_order,
|
|
bool is_tls_ring);
|
|
|
|
void
|
|
vn_ring_destroy(struct vn_ring *ring);
|
|
|
|
uint64_t
|
|
vn_ring_get_id(struct vn_ring *ring);
|
|
|
|
uint32_t
|
|
vn_ring_load_status(const struct vn_ring *ring);
|
|
|
|
void
|
|
vn_ring_unset_status_bits(struct vn_ring *ring, uint32_t mask);
|
|
|
|
bool
|
|
vn_ring_get_seqno_status(struct vn_ring *ring, uint32_t seqno);
|
|
|
|
void
|
|
vn_ring_wait_all(struct vn_ring *ring);
|
|
|
|
struct vn_ring_submit_command {
|
|
/* empty command implies errors */
|
|
struct vn_cs_encoder command;
|
|
struct vn_cs_encoder_buffer buffer;
|
|
/* non-zero implies waiting */
|
|
size_t reply_size;
|
|
|
|
/* when reply_size is non-zero, NULL can be returned on errors */
|
|
struct vn_renderer_shmem *reply_shmem;
|
|
struct vn_cs_decoder reply;
|
|
|
|
/* valid when ring submission succeeds */
|
|
bool ring_seqno_valid;
|
|
uint32_t ring_seqno;
|
|
};
|
|
|
|
static inline struct vn_cs_encoder *
|
|
vn_ring_submit_command_init(struct vn_ring *ring,
|
|
struct vn_ring_submit_command *submit,
|
|
void *cmd_data,
|
|
size_t cmd_size,
|
|
size_t reply_size)
|
|
{
|
|
submit->buffer = VN_CS_ENCODER_BUFFER_INITIALIZER(cmd_data);
|
|
submit->command = VN_CS_ENCODER_INITIALIZER(&submit->buffer, cmd_size);
|
|
|
|
submit->reply_size = reply_size;
|
|
submit->reply_shmem = NULL;
|
|
|
|
submit->ring_seqno_valid = false;
|
|
|
|
return &submit->command;
|
|
}
|
|
|
|
static inline struct vn_cs_decoder *
|
|
vn_ring_get_command_reply(struct vn_ring *ring,
|
|
struct vn_ring_submit_command *submit)
|
|
{
|
|
return submit->reply_shmem ? &submit->reply : NULL;
|
|
}
|
|
|
|
void
|
|
vn_ring_free_command_reply(struct vn_ring *ring,
|
|
struct vn_ring_submit_command *submit);
|
|
|
|
void
|
|
vn_ring_submit_command(struct vn_ring *ring,
|
|
struct vn_ring_submit_command *submit);
|
|
|
|
VkResult
|
|
vn_ring_submit_command_simple(struct vn_ring *ring,
|
|
const struct vn_cs_encoder *cs);
|
|
|
|
VkResult
|
|
vn_ring_submit_roundtrip(struct vn_ring *ring, uint64_t *roundtrip_seqno);
|
|
|
|
void
|
|
vn_ring_wait_roundtrip(struct vn_ring *ring, uint64_t roundtrip_seqno);
|
|
|
|
static inline void
|
|
vn_ring_roundtrip(struct vn_ring *ring)
|
|
{
|
|
uint64_t roundtrip_seqno;
|
|
if (vn_ring_submit_roundtrip(ring, &roundtrip_seqno) == VK_SUCCESS)
|
|
vn_ring_wait_roundtrip(ring, roundtrip_seqno);
|
|
}
|
|
|
|
#endif /* VN_RING_H */
|