/*
* This file is part of libplacebo.
*
* libplacebo is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* libplacebo is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with libplacebo. If not, see .
*/
#include
#include
#include "gpu.h"
#include
const struct pl_gpu_dummy_params pl_gpu_dummy_default_params = { PL_GPU_DUMMY_DEFAULTS };
static const struct pl_gpu_fns pl_fns_dummy;
struct priv {
struct pl_gpu_fns impl;
struct pl_gpu_dummy_params params;
};
pl_gpu pl_gpu_dummy_create(pl_log log, const struct pl_gpu_dummy_params *params)
{
params = PL_DEF(params, &pl_gpu_dummy_default_params);
struct pl_gpu_t *gpu = pl_zalloc_obj(NULL, gpu, struct priv);
gpu->log = log;
gpu->glsl = params->glsl;
gpu->limits = params->limits;
struct priv *p = PL_PRIV(gpu);
p->impl = pl_fns_dummy;
p->params = *params;
// Forcibly override these, because we know for sure what the values are
gpu->limits.align_tex_xfer_pitch = 1;
gpu->limits.align_tex_xfer_offset = 1;
gpu->limits.align_vertex_stride = 1;
// Set up the dummy formats, add one for each possible format type that we
// can represent on the host
PL_ARRAY(pl_fmt) formats = {0};
for (enum pl_fmt_type type = 1; type < PL_FMT_TYPE_COUNT; type++) {
for (int comps = 1; comps <= 4; comps++) {
for (int depth = 8; depth < 128; depth *= 2) {
if (type == PL_FMT_FLOAT && depth < 16)
continue;
static const char *cnames[] = {
[1] = "r",
[2] = "rg",
[3] = "rgb",
[4] = "rgba",
};
static const char *tnames[] = {
[PL_FMT_UNORM] = "",
[PL_FMT_SNORM] = "s",
[PL_FMT_UINT] = "u",
[PL_FMT_SINT] = "i",
[PL_FMT_FLOAT] = "f",
};
const char *tname = tnames[type];
if (type == PL_FMT_FLOAT && depth == 16)
tname = "hf";
struct pl_fmt_t *fmt = pl_alloc_ptr(gpu, fmt);
*fmt = (struct pl_fmt_t) {
.name = pl_asprintf(fmt, "%s%d%s", cnames[comps], depth, tname),
.type = type,
.num_components = comps,
.opaque = false,
.gatherable = true,
.internal_size = comps * depth / 8,
.texel_size = comps * depth / 8,
.texel_align = 1,
.caps = PL_FMT_CAP_SAMPLEABLE | PL_FMT_CAP_LINEAR |
PL_FMT_CAP_RENDERABLE | PL_FMT_CAP_BLENDABLE |
PL_FMT_CAP_VERTEX | PL_FMT_CAP_HOST_READABLE,
};
for (int i = 0; i < comps; i++) {
fmt->component_depth[i] = depth;
fmt->host_bits[i] = depth;
fmt->sample_order[i] = i;
}
if (gpu->glsl.compute)
fmt->caps |= PL_FMT_CAP_STORABLE;
if (gpu->limits.max_buffer_texels && gpu->limits.max_ubo_size)
fmt->caps |= PL_FMT_CAP_TEXEL_UNIFORM;
if (gpu->limits.max_buffer_texels && gpu->limits.max_ssbo_size)
fmt->caps |= PL_FMT_CAP_TEXEL_STORAGE;
fmt->glsl_type = pl_var_glsl_type_name(pl_var_from_fmt(fmt, ""));
fmt->glsl_format = pl_fmt_glsl_format(fmt, comps);
fmt->fourcc = pl_fmt_fourcc(fmt);
if (!fmt->glsl_format)
fmt->caps &= ~(PL_FMT_CAP_STORABLE | PL_FMT_CAP_TEXEL_STORAGE);
PL_ARRAY_APPEND(gpu, formats, fmt);
}
}
}
gpu->formats = formats.elem;
gpu->num_formats = formats.num;
return pl_gpu_finalize(gpu);
}
static void dumb_destroy(pl_gpu gpu)
{
pl_free((void *) gpu);
}
void pl_gpu_dummy_destroy(pl_gpu *gpu)
{
pl_gpu_destroy(*gpu);
*gpu = NULL;
}
struct buf_priv {
uint8_t *data;
};
static pl_buf dumb_buf_create(pl_gpu gpu, const struct pl_buf_params *params)
{
struct pl_buf_t *buf = pl_zalloc_obj(NULL, buf, struct buf_priv);
buf->params = *params;
buf->params.initial_data = NULL;
struct buf_priv *p = PL_PRIV(buf);
p->data = malloc(params->size);
if (!p->data) {
PL_ERR(gpu, "Failed allocating memory for dummy buffer!");
pl_free(buf);
return NULL;
}
if (params->initial_data)
memcpy(p->data, params->initial_data, params->size);
if (params->host_mapped)
buf->data = p->data;
return buf;
}
static void dumb_buf_destroy(pl_gpu gpu, pl_buf buf)
{
struct buf_priv *p = PL_PRIV(buf);
free(p->data);
pl_free((void *) buf);
}
uint8_t *pl_buf_dummy_data(pl_buf buf)
{
struct buf_priv *p = PL_PRIV(buf);
return p->data;
}
static void dumb_buf_write(pl_gpu gpu, pl_buf buf, size_t buf_offset,
const void *data, size_t size)
{
struct buf_priv *p = PL_PRIV(buf);
memcpy(p->data + buf_offset, data, size);
}
static bool dumb_buf_read(pl_gpu gpu, pl_buf buf, size_t buf_offset,
void *dest, size_t size)
{
struct buf_priv *p = PL_PRIV(buf);
memcpy(dest, p->data + buf_offset, size);
return true;
}
static void dumb_buf_copy(pl_gpu gpu, pl_buf dst, size_t dst_offset,
pl_buf src, size_t src_offset, size_t size)
{
struct buf_priv *dstp = PL_PRIV(dst);
struct buf_priv *srcp = PL_PRIV(src);
memcpy(dstp->data + dst_offset, srcp->data + src_offset, size);
}
struct tex_priv {
void *data;
};
static size_t tex_size(pl_gpu gpu, pl_tex tex)
{
size_t size = tex->params.format->texel_size * tex->params.w;
size *= PL_DEF(tex->params.h, 1);
size *= PL_DEF(tex->params.d, 1);
return size;
}
static pl_tex dumb_tex_create(pl_gpu gpu, const struct pl_tex_params *params)
{
struct pl_tex_t *tex = pl_zalloc_obj(NULL, tex, void *);
tex->params = *params;
tex->params.initial_data = NULL;
struct tex_priv *p = PL_PRIV(tex);
p->data = malloc(tex_size(gpu, tex));
if (!p->data) {
PL_ERR(gpu, "Failed allocating memory for dummy texture!");
pl_free(tex);
return NULL;
}
if (params->initial_data)
memcpy(p->data, params->initial_data, tex_size(gpu, tex));
return tex;
}
pl_tex pl_tex_dummy_create(pl_gpu gpu, const struct pl_tex_dummy_params *params)
{
// Only do minimal sanity checking, since this is just a dummy texture
pl_assert(params->format && params->w >= 0 && params->h >= 0 && params->d >= 0);
struct pl_tex_t *tex = pl_zalloc_obj(NULL, tex, struct tex_priv);
tex->sampler_type = params->sampler_type;
tex->params = (struct pl_tex_params) {
.w = params->w,
.h = params->h,
.d = params->d,
.format = params->format,
.sampleable = true,
.user_data = params->user_data,
};
return tex;
}
static void dumb_tex_destroy(pl_gpu gpu, pl_tex tex)
{
struct tex_priv *p = PL_PRIV(tex);
if (p->data)
free(p->data);
pl_free((void *) tex);
}
uint8_t *pl_tex_dummy_data(pl_tex tex)
{
struct tex_priv *p = PL_PRIV(tex);
return p->data;
}
static bool dumb_tex_upload(pl_gpu gpu, const struct pl_tex_transfer_params *params)
{
pl_tex tex = params->tex;
struct tex_priv *p = PL_PRIV(tex);
pl_assert(p->data);
const uint8_t *src = params->ptr;
uint8_t *dst = p->data;
if (params->buf) {
struct buf_priv *bufp = PL_PRIV(params->buf);
src = (uint8_t *) bufp->data + params->buf_offset;
}
size_t texel_size = tex->params.format->texel_size;
size_t row_size = pl_rect_w(params->rc) * texel_size;
for (int z = params->rc.z0; z < params->rc.z1; z++) {
size_t src_plane = z * params->depth_pitch;
size_t dst_plane = z * tex->params.h * tex->params.w * texel_size;
for (int y = params->rc.y0; y < params->rc.y1; y++) {
size_t src_row = src_plane + y * params->row_pitch;
size_t dst_row = dst_plane + y * tex->params.w * texel_size;
size_t pos = params->rc.x0 * texel_size;
memcpy(&dst[dst_row + pos], &src[src_row + pos], row_size);
}
}
return true;
}
static bool dumb_tex_download(pl_gpu gpu, const struct pl_tex_transfer_params *params)
{
pl_tex tex = params->tex;
struct tex_priv *p = PL_PRIV(tex);
pl_assert(p->data);
const uint8_t *src = p->data;
uint8_t *dst = params->ptr;
if (params->buf) {
struct buf_priv *bufp = PL_PRIV(params->buf);
dst = (uint8_t *) bufp->data + params->buf_offset;
}
size_t texel_size = tex->params.format->texel_size;
size_t row_size = pl_rect_w(params->rc) * texel_size;
for (int z = params->rc.z0; z < params->rc.z1; z++) {
size_t src_plane = z * tex->params.h * tex->params.w * texel_size;
size_t dst_plane = z * params->depth_pitch;
for (int y = params->rc.y0; y < params->rc.y1; y++) {
size_t src_row = src_plane + y * tex->params.w * texel_size;
size_t dst_row = dst_plane + y * params->row_pitch;
size_t pos = params->rc.x0 * texel_size;
memcpy(&dst[dst_row + pos], &src[src_row + pos], row_size);
}
}
return true;
}
static int dumb_desc_namespace(pl_gpu gpu, enum pl_desc_type type)
{
return 0; // safest behavior: never alias bindings
}
static pl_pass dumb_pass_create(pl_gpu gpu, const struct pl_pass_params *params)
{
PL_ERR(gpu, "Creating render passes is not supported for dummy GPUs");
return NULL;
}
static void dumb_gpu_finish(pl_gpu gpu)
{
// no-op
}
static const struct pl_gpu_fns pl_fns_dummy = {
.destroy = dumb_destroy,
.buf_create = dumb_buf_create,
.buf_destroy = dumb_buf_destroy,
.buf_write = dumb_buf_write,
.buf_read = dumb_buf_read,
.buf_copy = dumb_buf_copy,
.tex_create = dumb_tex_create,
.tex_destroy = dumb_tex_destroy,
.tex_upload = dumb_tex_upload,
.tex_download = dumb_tex_download,
.desc_namespace = dumb_desc_namespace,
.pass_create = dumb_pass_create,
.gpu_finish = dumb_gpu_finish,
};