mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2024-11-30 08:00:51 +00:00
260a10d75c
It's not fully compliant yet, but it sorta kinda maybe worked on my pinebook (still slow, but no hardware vulkan :( )
1548 lines
41 KiB
C
1548 lines
41 KiB
C
/*
|
|
vkparse.c
|
|
|
|
Parser for scripted vulkan structs
|
|
|
|
Copyright (C) 2020 Bill Currie <bill@taniwha.org>
|
|
|
|
This program is free software; you can redistribute it and/or
|
|
modify it under the terms of the GNU General Public License
|
|
as published by the Free Software Foundation; either version 2
|
|
of the License, or (at your option) any later version.
|
|
|
|
This program 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 General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to:
|
|
|
|
Free Software Foundation, Inc.
|
|
59 Temple Place - Suite 330
|
|
Boston, MA 02111-1307, USA
|
|
|
|
*/
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
#include <string.h>
|
|
|
|
#include "QF/cmem.h"
|
|
#include "QF/cvar.h"
|
|
#include "QF/dstring.h"
|
|
#include "QF/hash.h"
|
|
#include "QF/mathlib.h"
|
|
#include "QF/va.h"
|
|
|
|
#include "QF/Vulkan/debug.h"
|
|
#include "QF/Vulkan/device.h"
|
|
#include "QF/Vulkan/instance.h"
|
|
#include "QF/Vulkan/image.h"
|
|
#include "QF/Vulkan/pipeline.h"
|
|
#include "QF/Vulkan/render.h"
|
|
#include "QF/Vulkan/shader.h"
|
|
|
|
#include "vid_vulkan.h"
|
|
|
|
#include "vkparse.h"
|
|
#include "libs/video/renderer/vulkan/vkparse.hinc"
|
|
|
|
typedef struct parseres_s {
|
|
const char *name;
|
|
plfield_t *field;
|
|
size_t offset;
|
|
} parseres_t;
|
|
|
|
typedef struct parseref_s {
|
|
const char *name;
|
|
plparser_t parse;
|
|
size_t size;
|
|
} parserref_t;
|
|
|
|
typedef struct handleref_s {
|
|
char *name;
|
|
uint64_t handle;
|
|
} handleref_t;
|
|
|
|
static void flag_or (const exprval_t *val1, const exprval_t *val2,
|
|
exprval_t *result, exprctx_t *ctx)
|
|
{
|
|
*(int *) (result->value) = *(int *) (val1->value) | *(int *) (val2->value);
|
|
}
|
|
|
|
static void flag_and (const exprval_t *val1, const exprval_t *val2,
|
|
exprval_t *result, exprctx_t *ctx)
|
|
{
|
|
*(int *) (result->value) = *(int *) (val1->value) & *(int *) (val2->value);
|
|
}
|
|
|
|
static void flag_cast_int (const exprval_t *val1, const exprval_t *val2,
|
|
exprval_t *result, exprctx_t *ctx)
|
|
{
|
|
// FIXME should check value is valid
|
|
*(int *) (result->value) = *(int *) (val2->value);
|
|
}
|
|
|
|
static void flag_not (const exprval_t *val, exprval_t *result, exprctx_t *ctx)
|
|
{
|
|
*(int *) (result->value) = ~(*(int *) (val->value));
|
|
}
|
|
|
|
binop_t flag_binops[] = {
|
|
{ '|', 0, 0, flag_or },
|
|
{ '&', 0, 0, flag_and },
|
|
{ '=', &cexpr_int, 0, flag_cast_int },
|
|
{ '=', &cexpr_plitem, 0, cexpr_cast_plitem },
|
|
{}
|
|
};
|
|
|
|
binop_t enum_binops[] = {
|
|
{ '=', &cexpr_plitem, 0, cexpr_cast_plitem },
|
|
{}
|
|
};
|
|
|
|
unop_t flag_unops[] = {
|
|
{ '~', 0, flag_not },
|
|
{}
|
|
};
|
|
|
|
typedef struct parse_single_s {
|
|
pltype_t type;
|
|
size_t stride;
|
|
plparser_t parser;
|
|
void *data;
|
|
size_t value_offset;
|
|
} parse_single_t;
|
|
|
|
typedef struct parse_array_s {
|
|
pltype_t type;
|
|
size_t stride;
|
|
plparser_t parser;
|
|
void *data;
|
|
size_t value_offset;
|
|
size_t size_offset;
|
|
} parse_array_t;
|
|
|
|
typedef struct parse_fixed_array_s {
|
|
pltype_t type;
|
|
size_t stride;
|
|
plparser_t parser;
|
|
void *data;
|
|
size_t size;
|
|
} parse_fixed_array_t;
|
|
|
|
typedef struct parse_data_s {
|
|
size_t value_offset;
|
|
size_t size_offset;
|
|
} parse_data_t;
|
|
|
|
typedef struct parse_string_s {
|
|
size_t value_offset;
|
|
} parse_string_t;
|
|
|
|
typedef struct parse_custom_s {
|
|
int (*parse) (const plitem_t *item, void **data,
|
|
plitem_t *messages, parsectx_t *context);
|
|
size_t *offsets;
|
|
size_t num_offsets;
|
|
} parse_custom_t;
|
|
|
|
static plfield_t *__attribute__((used))
|
|
find_field (plfield_t *fields, const char *field_name,
|
|
const plitem_t *item, plitem_t *messages)
|
|
{
|
|
for (plfield_t *f = fields; f->name; f++) {
|
|
if (strcmp (f->name, field_name) == 0) {
|
|
return f;
|
|
}
|
|
}
|
|
PL_Message (messages, item, "error: unknown field '%s'", field_name);
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
parse_basic (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
int ret = 1;
|
|
parsectx_t *pctx = context;
|
|
__auto_type etype = (exprtype_t *) field->data;
|
|
exprctx_t ectx = *pctx->ectx;
|
|
exprval_t result = { etype, data };
|
|
ectx.result = &result;
|
|
ectx.item = item;
|
|
const char *valstr = PL_String (item);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, "parse_basic: %s %zd %d %p %p: %s\n",
|
|
// field->name, field->offset, field->type, field->parser,
|
|
// field->data, valstr);
|
|
if (strcmp (valstr, "VK_SUBPASS_EXTERNAL") == 0) {
|
|
//FIXME handle subpass in a separate parser?
|
|
*(uint32_t *) data = VK_SUBPASS_EXTERNAL;
|
|
} else {
|
|
ret = !cexpr_eval_string (valstr, &ectx);
|
|
if (!ret) {
|
|
PL_Message (messages, item, "error parsing %s: %s",
|
|
field->name, valstr);
|
|
}
|
|
}
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, " %x\n", *(uint32_t *)data);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
parse_int32_t (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
int ret = 1;
|
|
int val = 0;
|
|
parsectx_t *pctx = context;
|
|
exprval_t result = { &cexpr_int, &val };
|
|
exprctx_t ectx = *pctx->ectx;
|
|
ectx.result = &result;
|
|
ectx.item = item;
|
|
const char *valstr = PL_String (item);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse,
|
|
// "parse_int32_t: %s %zd %d %p %p %s\n",
|
|
// field->name, field->offset, field->type, field->parser,
|
|
// field->data, valstr);
|
|
ret = !cexpr_eval_string (valstr, &ectx);
|
|
if (!ret) {
|
|
PL_Message (messages, item, "error parsing %s: %s",
|
|
field->name, valstr);
|
|
}
|
|
*(int32_t *) data = val;
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, " %d\n", *(int32_t *)data);
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
parse_uint32_t (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
int ret = 1;
|
|
// use size_t (and cexpr_size_t) for val so references to array sizes
|
|
// can be used
|
|
size_t val = 0;
|
|
parsectx_t *pctx = context;
|
|
exprval_t result = { &cexpr_size_t, &val };
|
|
exprctx_t ectx = *pctx->ectx;
|
|
ectx.result = &result;
|
|
ectx.item = item;
|
|
const char *valstr = PL_String (item);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, "parse_uint32_t: %s %zd %d %p %p: %s\n",
|
|
// field->name, field->offset, field->type, field->parser,
|
|
// field->data, valstr);
|
|
if (strcmp (valstr, "VK_SUBPASS_EXTERNAL") == 0) {
|
|
//FIXME handle subpass in a separate parser?
|
|
*(uint32_t *) data = VK_SUBPASS_EXTERNAL;
|
|
} else {
|
|
//Sys_MaskPrintf (SYS_vulkan_parse,
|
|
// "parse_uint32_t: %s %zd %d %p %p %s\n",
|
|
// field->name, field->offset, field->type, field->parser,
|
|
// field->data, valstr);
|
|
ret = !cexpr_eval_string (valstr, &ectx);
|
|
if (!ret) {
|
|
PL_Message (messages, item, "error parsing %s: %s",
|
|
field->name, valstr);
|
|
}
|
|
*(uint32_t *) data = val;
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, " %d\n", *(uint32_t *)data);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
parse_enum (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
int ret = 1;
|
|
__auto_type enm = (exprenum_t *) field->data;
|
|
parsectx_t *pctx = context;
|
|
exprctx_t ectx = *pctx->ectx;
|
|
exprval_t result = { enm->type, data };
|
|
ectx.parent = pctx->ectx;
|
|
ectx.symtab = enm->symtab;
|
|
ectx.result = &result;
|
|
const char *valstr = PL_String (item);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, "parse_enum: %s %zd %d %p %p %s\n",
|
|
// field->name, field->offset, field->type, field->parser,
|
|
// field->data, valstr);
|
|
ret = !cexpr_parse_enum (enm, valstr, &ectx, data);
|
|
if (!ret) {
|
|
PL_Message (messages, item, "error parsing enum: %s", valstr);
|
|
}
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, " %d\n", *(int *)data);
|
|
return ret;
|
|
}
|
|
|
|
static const plitem_t *
|
|
parse_reference (const plitem_t *item, const char *type, plitem_t *messages,
|
|
parsectx_t *pctx)
|
|
{
|
|
exprctx_t ectx = *pctx->ectx;
|
|
plitem_t *refItem = 0;
|
|
exprval_t result = { &cexpr_plitem, &refItem };
|
|
ectx.result = &result;
|
|
ectx.item = item;
|
|
const char *name = PL_String (item);
|
|
if (cexpr_eval_string (name, &ectx)) {
|
|
PL_Message (messages, item, "not a %s reference", type);
|
|
return 0;
|
|
}
|
|
return refItem;
|
|
}
|
|
|
|
static void *
|
|
vkparse_alloc (void *context, size_t size)
|
|
{
|
|
parsectx_t *pctx = context;
|
|
return cmemalloc (pctx->ectx->memsuper, size);
|
|
}
|
|
|
|
static int
|
|
parse_ignore (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
static int __attribute__((used))
|
|
parse_labeledsingle (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
__auto_type single = (parse_single_t *) field->data;
|
|
void *flddata = (byte *)data + single->value_offset;
|
|
|
|
//Sys_MaskPrintf (SYS_vulkan_parse,"parse_labeledsingle: %s %zd %d %p %p\n",
|
|
// field->name, field->offset,
|
|
// field->type, field->parser, field->data);
|
|
|
|
const char *key = PL_KeyAtIndex (item, 0);
|
|
if (!key) {
|
|
PL_Message (messages, item, "missing item");
|
|
return 0;
|
|
}
|
|
item = PL_ObjectForKey (item, key);
|
|
|
|
if (!PL_CheckType (single->type, PL_Type (item))) {
|
|
PL_TypeMismatch (messages, item, field->name, single->type,
|
|
PL_Type (item));
|
|
return 0;
|
|
}
|
|
|
|
plfield_t f = { field->name, 0, single->type, single->parser, 0 };
|
|
void *value = vkparse_alloc (context, single->stride);
|
|
memset (value, 0, single->stride);
|
|
if (!single->parser (&f, item, value, messages, context)) {
|
|
return 0;
|
|
}
|
|
|
|
*(void **) flddata = value;
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
parse_single (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
__auto_type single = (parse_single_t *) field->data;
|
|
void *flddata = (byte *)data + single->value_offset;
|
|
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, "parse_single: %s %zd %d %p %p\n",
|
|
// field->name, field->offset,
|
|
// field->type, field->parser, field->data);
|
|
|
|
if (!PL_CheckType (single->type, PL_Type (item))) {
|
|
PL_TypeMismatch (messages, item, field->name, single->type,
|
|
PL_Type (item));
|
|
return 0;
|
|
}
|
|
|
|
plfield_t f = { field->name, 0, single->type, single->parser, 0 };
|
|
void *value = vkparse_alloc (context, single->stride);
|
|
memset (value, 0, single->stride);
|
|
if (!single->parser (&f, item, value, messages, context)) {
|
|
return 0;
|
|
}
|
|
|
|
*(void **) flddata = value;
|
|
return 1;
|
|
}
|
|
|
|
static int __attribute__((used))
|
|
parse_labeledarray (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
__auto_type array = (parse_array_t *) field->data;
|
|
__auto_type value = (void **) ((byte *)data + array->value_offset);
|
|
__auto_type size = (uint32_t *) ((byte *)data + array->size_offset);
|
|
|
|
plelement_t element = {
|
|
array->type,
|
|
array->stride,
|
|
vkparse_alloc,
|
|
array->parser,
|
|
array->data,
|
|
};
|
|
plfield_t f = { 0, 0, 0, 0, &element };
|
|
|
|
typedef struct arr_s DARRAY_TYPE(byte) arr_t;
|
|
arr_t *arr;
|
|
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, "parse_array: %s %zd %d %p %p %p\n",
|
|
// field->name, field->offset, field->type, field->parser,
|
|
// field->data, data);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, " %d %zd %p %zd %zd\n", array->type,
|
|
// array->stride, array->parser, array->value_offset,
|
|
// array->size_offset);
|
|
if (!PL_ParseLabeledArray (&f, item, &arr, messages, context)) {
|
|
return 0;
|
|
}
|
|
*value = vkparse_alloc (context, array->stride * arr->size);
|
|
memcpy (*value, arr->a, array->stride * arr->size);
|
|
if ((void *) size >= data) {
|
|
*size = arr->size;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
parse_array (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
__auto_type array = (parse_array_t *) field->data;
|
|
__auto_type value = (void **) ((byte *)data + array->value_offset);
|
|
__auto_type size = (uint32_t *) ((byte *)data + array->size_offset);
|
|
|
|
plelement_t element = {
|
|
array->type,
|
|
array->stride,
|
|
vkparse_alloc,
|
|
array->parser,
|
|
array->data,
|
|
};
|
|
plfield_t f = { 0, 0, 0, 0, &element };
|
|
|
|
typedef struct arr_s DARRAY_TYPE(byte) arr_t;
|
|
arr_t *arr;
|
|
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, "parse_array: %s %zd %d %p %p %p\n",
|
|
// field->name, field->offset, field->type, field->parser,
|
|
// field->data, data);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, " %d %zd %p %zd %zd\n", array->type,
|
|
// array->stride, array->parser, array->value_offset,
|
|
// array->size_offset);
|
|
if (!PL_ParseArray (&f, item, &arr, messages, context)) {
|
|
return 0;
|
|
}
|
|
*value = vkparse_alloc (context, array->stride * arr->size);
|
|
memcpy (*value, arr->a, array->stride * arr->size);
|
|
if ((void *) size >= data) {
|
|
*size = arr->size;
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
static int __attribute__((used))
|
|
parse_fixed_array (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
__auto_type array = (parse_fixed_array_t *) field->data;
|
|
|
|
plelement_t element = {
|
|
array->type,
|
|
array->stride,
|
|
vkparse_alloc,
|
|
array->parser,
|
|
array->data,
|
|
};
|
|
plfield_t f = { 0, 0, 0, 0, &element };
|
|
|
|
typedef struct arr_s DARRAY_TYPE(byte) arr_t;
|
|
arr_t *arr;
|
|
|
|
if (!PL_ParseArray (&f, item, &arr, messages, context)) {
|
|
return 0;
|
|
}
|
|
memset (data, 0, array->stride * array->size);
|
|
size_t size = min (array->size, arr->size);
|
|
memcpy (data, arr->a, array->stride * size);
|
|
return 1;
|
|
}
|
|
|
|
static char *
|
|
vkstrdup (parsectx_t *context, const char *str)
|
|
{
|
|
size_t len = strlen (str) + 1;
|
|
char *dup = vkparse_alloc (context, len);
|
|
memcpy (dup, str, len);
|
|
return dup;
|
|
}
|
|
|
|
static __attribute__((used)) parse_string_t parse_string_array = { 0 };
|
|
|
|
static int
|
|
parse_string (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
__auto_type string = (parse_string_t *) field->data;
|
|
__auto_type value = (char **) ((byte *)data + string->value_offset);
|
|
|
|
const char *str = PL_String (item);
|
|
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, "parse_string: %s %zd %d %p %p %p\n",
|
|
// field->name, field->offset, field->type, field->parser,
|
|
// field->data, data);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, " %zd\n", string->value_offset);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, " %s\n", str);
|
|
|
|
*value = vkstrdup (context, str);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
parse_custom (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
__auto_type custom = (parse_custom_t *) field->data;
|
|
void **offsets = alloca (custom->num_offsets * sizeof (void *));
|
|
for (size_t i = 0; i < custom->num_offsets; i++) {
|
|
offsets[i] = data + custom->offsets[i];
|
|
}
|
|
return custom->parse (item, offsets, messages, context);
|
|
}
|
|
|
|
static int
|
|
parse_inherit (const plfield_t *field, const plitem_t *item,
|
|
void *data, plitem_t *messages, void *context)
|
|
{
|
|
parsectx_t *pctx = context;
|
|
exprctx_t ectx = *pctx->ectx;
|
|
plitem_t *inheritItem = 0;
|
|
exprval_t result = { &cexpr_plitem, &inheritItem };
|
|
ectx.result = &result;
|
|
ectx.item = item;
|
|
const char *inheritstr = PL_String (item);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, "parse_inherit: %s\n", inheritstr);
|
|
int ret = !cexpr_eval_string (inheritstr, &ectx);
|
|
if (ret) {
|
|
ret = PL_ParseStruct (field->data, inheritItem, data, messages,
|
|
context);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static hashtab_t *parser_table;
|
|
|
|
static int
|
|
parse_next (const plfield_t *field, const plitem_t *item, void *data,
|
|
plitem_t *messages, void *context)
|
|
{
|
|
const char *type_name = PL_String (PL_ObjectAtIndex (item, 0));
|
|
plitem_t *next_def = PL_ObjectAtIndex (item, 1);
|
|
|
|
if (!type_name || PL_Type (next_def) != QFDictionary) {
|
|
PL_Message (messages, item, "invalid @next");
|
|
return 0;
|
|
}
|
|
parserref_t *parser = Hash_Find (parser_table, type_name);
|
|
if (!parser) {
|
|
PL_Message (messages, item, "Invalid type for @next: %s", type_name);
|
|
return 0;
|
|
}
|
|
void *data_ptr = vkparse_alloc (context, parser->size);
|
|
memset (data_ptr, 0, parser->size);
|
|
if (!parser->parse (field, next_def, data_ptr, messages, context)) {
|
|
return 0;
|
|
}
|
|
*(void **) data = data_ptr;
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
parse_RGBA (const plitem_t *item, void **data,
|
|
plitem_t *messages, parsectx_t *context)
|
|
{
|
|
int ret = 1;
|
|
exprctx_t ectx = *context->ectx;
|
|
exprval_t result = { &cexpr_vector, data[0] };
|
|
ectx.result = &result;
|
|
ectx.item = item;
|
|
const char *valstr = PL_String (item);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, "parse_RGBA: %s\n", valstr);
|
|
ret = !cexpr_eval_string (valstr, &ectx);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse, " "VEC4F_FMT"\n",
|
|
// VEC4_EXP (*(vec4f_t *)data[0]));
|
|
return ret;
|
|
}
|
|
|
|
uint64_t
|
|
QFV_GetHandle (hashtab_t *tab, const char *name)
|
|
{
|
|
handleref_t *hr = Hash_Find (tab, name);
|
|
if (hr) {
|
|
return hr->handle;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
QFV_AddHandle (hashtab_t *tab, const char *name, uint64_t handle)
|
|
{
|
|
handleref_t *hr = malloc (sizeof (handleref_t));
|
|
hr->name = strdup (name);
|
|
hr->handle = handle;
|
|
Hash_Add (tab, hr);
|
|
}
|
|
|
|
static int
|
|
parse_VkShaderModule (const plitem_t *item, void **data,
|
|
plitem_t *messages, parsectx_t *pctx)
|
|
{
|
|
__auto_type handle = (VkShaderModule *) data[0];
|
|
vulkan_ctx_t *ctx = pctx->vctx;
|
|
qfv_device_t *device = ctx->device;
|
|
scriptctx_t *sctx = ctx->script_context;
|
|
|
|
const char *name = PL_String (item);
|
|
*handle = (VkShaderModule) QFV_GetHandle (sctx->shaderModules, name);
|
|
if (*handle) {
|
|
return 1;
|
|
}
|
|
qfvPushDebug (ctx, va (ctx->va_ctx, "parse_VkShaderModule: %d", PL_Line (item)));
|
|
*handle = QFV_CreateShaderModule (device, name);
|
|
qfvPopDebug (ctx);
|
|
if (!*handle) {
|
|
PL_Message (messages, item, "could not find shader %s", name);
|
|
return 0;
|
|
}
|
|
QFV_AddHandle (sctx->shaderModules, name, (uint64_t) *handle);
|
|
return 1;
|
|
}
|
|
|
|
exprtype_t VkImage_type = {
|
|
.name = "VkImage",
|
|
.size = sizeof (VkImage),
|
|
.binops = 0,
|
|
.unops = 0,
|
|
.data = 0
|
|
};
|
|
|
|
exprtype_t VkImageView_type = {
|
|
.name = "VkImageView",
|
|
.size = sizeof (VkImageView),
|
|
.binops = 0,
|
|
.unops = 0,
|
|
.data = 0
|
|
};
|
|
|
|
static const char *
|
|
handleref_getkey (const void *hr, void *unused)
|
|
{
|
|
return ((handleref_t *)hr)->name;
|
|
}
|
|
|
|
static void
|
|
handleref_free (void *hr, void *_ctx)
|
|
{
|
|
__auto_type handleref = (handleref_t *) hr;
|
|
free (handleref->name);
|
|
free (handleref);
|
|
}
|
|
|
|
static void
|
|
setLayout_free (void *hr, void *_ctx)
|
|
{
|
|
scriptctx_t *sctx = _ctx;
|
|
__auto_type handleref = (handleref_t *) hr;
|
|
__auto_type layout = (VkDescriptorSetLayout) handleref->handle;
|
|
__auto_type ctx = sctx->vctx;
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
|
|
if (layout) {
|
|
dfunc->vkDestroyDescriptorSetLayout (device->dev, layout, 0);
|
|
}
|
|
handleref_free (handleref, ctx);
|
|
}
|
|
|
|
static void
|
|
shaderModule_free (void *hr, void *_ctx)
|
|
{
|
|
scriptctx_t *sctx = _ctx;
|
|
__auto_type handleref = (handleref_t *) hr;
|
|
__auto_type module = (VkShaderModule) handleref->handle;
|
|
__auto_type ctx = sctx->vctx;
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
|
|
if (module) {
|
|
dfunc->vkDestroyShaderModule (device->dev, module, 0);
|
|
}
|
|
handleref_free (handleref, ctx);
|
|
}
|
|
|
|
static void
|
|
pipelineLayout_free (void *hr, void *_ctx)
|
|
{
|
|
scriptctx_t *sctx = _ctx;
|
|
__auto_type handleref = (handleref_t *) hr;
|
|
__auto_type layout = (VkPipelineLayout) handleref->handle;
|
|
__auto_type ctx = sctx->vctx;
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
|
|
if (layout) {
|
|
dfunc->vkDestroyPipelineLayout (device->dev, layout, 0);
|
|
};
|
|
handleref_free (handleref, ctx);
|
|
}
|
|
|
|
static void
|
|
descriptorPool_free (void *hr, void *_ctx)
|
|
{
|
|
scriptctx_t *sctx = _ctx;
|
|
__auto_type handleref = (handleref_t *) hr;
|
|
__auto_type pool = (VkDescriptorPool) handleref->handle;
|
|
__auto_type ctx = sctx->vctx;
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
|
|
if (pool) {
|
|
dfunc->vkDestroyDescriptorPool (device->dev, pool, 0);
|
|
};
|
|
handleref_free (handleref, ctx);
|
|
}
|
|
|
|
static void
|
|
sampler_free (void *hr, void *_ctx)
|
|
{
|
|
scriptctx_t *sctx = _ctx;
|
|
__auto_type handleref = (handleref_t *) hr;
|
|
__auto_type sampler = (VkSampler) handleref->handle;
|
|
__auto_type ctx = sctx->vctx;
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
|
|
if (sampler) {
|
|
dfunc->vkDestroySampler (device->dev, sampler, 0);
|
|
};
|
|
handleref_free (handleref, ctx);
|
|
}
|
|
|
|
static void
|
|
image_free (void *hr, void *_ctx)
|
|
{
|
|
scriptctx_t *sctx = _ctx;
|
|
__auto_type handleref = (handleref_t *) hr;
|
|
__auto_type image = (VkImage) handleref->handle;
|
|
__auto_type ctx = sctx->vctx;
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
|
|
if (image) {
|
|
dfunc->vkDestroyImage (device->dev, image, 0);
|
|
};
|
|
handleref_free (handleref, ctx);
|
|
}
|
|
|
|
static void
|
|
imageView_free (void *hr, void *_ctx)
|
|
{
|
|
scriptctx_t *sctx = _ctx;
|
|
__auto_type handleref = (handleref_t *) hr;
|
|
__auto_type imageView = (VkImageView) handleref->handle;
|
|
__auto_type ctx = sctx->vctx;
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
|
|
if (imageView) {
|
|
dfunc->vkDestroyImageView (device->dev, imageView, 0);
|
|
};
|
|
handleref_free (handleref, ctx);
|
|
}
|
|
|
|
static void
|
|
renderpass_free (void *hr, void *_ctx)
|
|
{
|
|
scriptctx_t *sctx = _ctx;
|
|
__auto_type handleref = (handleref_t *) hr;
|
|
__auto_type renderpass = (VkRenderPass) handleref->handle;
|
|
__auto_type ctx = sctx->vctx;
|
|
qfv_device_t *device = ctx->device;
|
|
qfv_devfuncs_t *dfunc = device->funcs;
|
|
|
|
if (renderpass) {
|
|
dfunc->vkDestroyRenderPass (device->dev, renderpass, 0);
|
|
};
|
|
handleref_free (handleref, ctx);
|
|
}
|
|
|
|
static hashtab_t *enum_symtab;
|
|
|
|
typedef struct data_array_s DARRAY_TYPE(byte) data_array_t;
|
|
static void
|
|
data_array (const exprval_t **params, exprval_t *result, exprctx_t *context)
|
|
{
|
|
size_t offset = 0;
|
|
// params are in reverse order, but this works for calculating the size
|
|
// of the buffer
|
|
for (const exprval_t **param = params; *param; param++) {
|
|
offset += (*param)->type->size;
|
|
}
|
|
__auto_type data = DARRAY_ALLOCFIXED_OBJ (data_array_t, offset,
|
|
cmemalloc, context->memsuper);
|
|
for (const exprval_t **param = params; *param; param++) {
|
|
size_t size = (*param)->type->size;
|
|
// pre-decrement offset because params are in reverse order
|
|
offset -= size;
|
|
memcpy (data->a + offset, (*param)->value, size);
|
|
}
|
|
*(data_array_t **) result->value = data;
|
|
}
|
|
|
|
static exprtype_t data_array_type = {
|
|
.name = "array",
|
|
.size = sizeof (data_array_t *),
|
|
.binops = 0,
|
|
.unops = 0,
|
|
};
|
|
static exprfunc_t data_array_func[] = {
|
|
{ &data_array_type, -1, 0, data_array },
|
|
{}
|
|
};
|
|
static exprsym_t data_array_symbols[] = {
|
|
{ "array", &cexpr_function, data_array_func },
|
|
{}
|
|
};
|
|
static exprtab_t data_array_symtab = {
|
|
data_array_symbols,
|
|
};
|
|
|
|
static int
|
|
parse_specialization_data (const plitem_t *item, void **data,
|
|
plitem_t *messages, parsectx_t *pctx)
|
|
{
|
|
size_t *size_ptr = (size_t *) data[0];
|
|
void **data_ptr = (void **) data[1];
|
|
|
|
if (PL_Type (item) == QFBinary) {
|
|
const void *bindata = PL_BinaryData (item);
|
|
size_t binsize = PL_BinarySize (item);
|
|
|
|
*data_ptr = vkparse_alloc (pctx, binsize);
|
|
memcpy (*data_ptr, bindata, binsize);
|
|
*size_ptr = binsize;
|
|
return 1;
|
|
}
|
|
|
|
data_array_t *da= 0;
|
|
exprctx_t ectx = *pctx->ectx;
|
|
exprval_t result = { &data_array_type, &da };
|
|
ectx.parent = pctx->ectx;
|
|
ectx.symtab = &data_array_symtab;
|
|
ectx.result = &result;
|
|
ectx.item = item;
|
|
const char *valstr = PL_String (item);
|
|
//Sys_MaskPrintf (SYS_vulkan_parse,
|
|
// "parse_specialization_data: %s %zd %d %p %p %s\n",
|
|
// field->name, field->offset, field->type, field->parser,
|
|
// field->data, valstr);
|
|
int ret = !cexpr_eval_string (valstr, &ectx);
|
|
if (!ret) {
|
|
PL_Message (messages, item, "error parsing specialization data: %s",
|
|
valstr);
|
|
} else {
|
|
*size_ptr = da->size;
|
|
*data_ptr = da->a;
|
|
//for (size_t i = 0; i < da->size; i++) {
|
|
// Sys_Printf (" %02x", da->a[i]);
|
|
//}
|
|
//Sys_Printf ("\n");
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static int
|
|
parse_task_function (const plitem_t *item, void **data,
|
|
plitem_t *messages, parsectx_t *pctx)
|
|
{
|
|
qfv_renderctx_t *rctx = pctx->data;
|
|
const char *fname = PL_String (item);
|
|
exprsym_t *fsym = Hash_Find (rctx->task_functions.tab, fname);
|
|
|
|
if (!fsym) {
|
|
PL_Message (messages, item, "undefined task function %s", fname);
|
|
return 0;
|
|
}
|
|
if (fsym->type != &cexpr_function) {
|
|
PL_Message (messages, item, "not a function type %s", fname);
|
|
return 0;
|
|
}
|
|
exprfunc_t *func;
|
|
for (func = fsym->value; func->func; func++) {
|
|
if (!func->result) {
|
|
break;
|
|
}
|
|
}
|
|
if (!func->func) {
|
|
PL_Message (messages, item, "%s does not have a void implementation",
|
|
fname);
|
|
return 0;
|
|
}
|
|
size_t size = func->num_params * sizeof (exprval_t);
|
|
size += func->num_params * sizeof (exprval_t *);
|
|
size_t base = size;
|
|
for (int i = 0; i < func->num_params; i++) {
|
|
exprtype_t *type = func->param_types[i];
|
|
size = ((size + type->size - 1) & ~(type->size - 1));
|
|
if (i == 0) {
|
|
base = size;
|
|
}
|
|
}
|
|
exprval_t **param_ptrs = vkparse_alloc (pctx, size);
|
|
exprval_t *params = (exprval_t *) ¶m_ptrs[func->num_params];
|
|
byte *param_data = (byte *) param_ptrs + base;
|
|
memset (params, 0, size);
|
|
size_t offs = 0;
|
|
for (int i = 0; i < func->num_params; i++) {
|
|
exprtype_t *type = func->param_types[i];
|
|
param_ptrs[i] = ¶ms[i];
|
|
params[i] = (exprval_t) {
|
|
.type = type,
|
|
.value = param_data + offs,
|
|
};
|
|
offs = ((offs + type->size - 1) & ~(type->size - 1));
|
|
offs += type->size;
|
|
}
|
|
*(exprfunc_t **) data[0] = func;
|
|
*(exprval_t ***) data[1] = param_ptrs;
|
|
*(void **) data[2] = param_data;
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
parse_task_params (const plitem_t *item, void **data,
|
|
plitem_t *messages, parsectx_t *pctx)
|
|
{
|
|
exprfunc_t *func = *(exprfunc_t **) data[0];
|
|
exprval_t **params = *(exprval_t ***) data[1];
|
|
if (!func) {
|
|
PL_Message (messages, item, "task function not set");
|
|
return 0;
|
|
}
|
|
if (PL_A_NumObjects (item) != func->num_params) {
|
|
PL_Message (messages, item, "incorrect number of parameters");
|
|
return 0;
|
|
}
|
|
for (int i = 0; i < func->num_params; i++) {
|
|
const char *paramstr = PL_String (PL_ObjectAtIndex (item, i));
|
|
exprval_t *param = params[func->num_params - i - 1];
|
|
exprctx_t ectx = *pctx->ectx;
|
|
if (param->type->data) {
|
|
ectx.parent = pctx->ectx;
|
|
ectx.symtab = ((exprenum_t *) param->type->data)->symtab;
|
|
}
|
|
// cexpr params are in reverse order
|
|
ectx.result = param;
|
|
|
|
if (cexpr_eval_string (paramstr, &ectx)) {
|
|
PL_Message (messages, item, "error parsing param %d", i);
|
|
return 0;
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
#include "libs/video/renderer/vulkan/vkparse.cinc"
|
|
|
|
static exprsym_t qfv_renderframeset_t_symbols[] = {
|
|
{"size", &cexpr_size_t, (void *)field_offset (qfv_renderframeset_t, size)},
|
|
{ }
|
|
};
|
|
static exprtab_t qfv_renderframeset_t_symtab = {
|
|
qfv_renderframeset_t_symbols,
|
|
};
|
|
exprtype_t qfv_renderframeset_t_type = {
|
|
.name = "frameset",
|
|
.size = sizeof (qfv_renderframeset_t *),
|
|
.binops = cexpr_struct_binops,
|
|
.unops = 0,
|
|
.data = &qfv_renderframeset_t_symtab,
|
|
};
|
|
|
|
static hashtab_t *
|
|
handlref_symtab (void (*free_func)(void*,void*), scriptctx_t *sctx)
|
|
{
|
|
return Hash_NewTable (23, handleref_getkey, free_func, sctx,
|
|
&sctx->hashctx);
|
|
}
|
|
|
|
static const char *
|
|
enum_symtab_getkey (const void *e, void *unused)
|
|
{
|
|
__auto_type enm = (const exprenum_t *) e;
|
|
return enm->type->name;
|
|
}
|
|
|
|
static const char *
|
|
parser_getkey (const void *e, void *unused)
|
|
{
|
|
__auto_type parser = (const parserref_t *) e;
|
|
return parser->name;
|
|
}
|
|
|
|
static exprtab_t root_symtab = {
|
|
.symbols = cexpr_lib_symbols,
|
|
};
|
|
|
|
static void
|
|
root_symtab_shutdown (void *data)
|
|
{
|
|
Hash_DelTable (root_symtab.tab);
|
|
}
|
|
|
|
static void __attribute__((constructor))
|
|
root_symtab_init (void)
|
|
{
|
|
// using a null hashctx here is safe because this function is run before
|
|
// main and thus before any possibility of threading.
|
|
exprctx_t root_context = { .symtab = &root_symtab };
|
|
cexpr_init_symtab (&root_symtab, &root_context);
|
|
Sys_RegisterShutdown (root_symtab_shutdown, 0);
|
|
}
|
|
|
|
exprenum_t *
|
|
QFV_GetEnum (const char *name)
|
|
{
|
|
return Hash_Find (enum_symtab, name);
|
|
}
|
|
|
|
static int
|
|
parse_object (vulkan_ctx_t *ctx, memsuper_t *memsuper, plitem_t *plist,
|
|
plparser_t parser, void *object, plitem_t *properties)
|
|
{
|
|
scriptctx_t *sctx = ctx->script_context;
|
|
plitem_t *messages = PL_NewArray ();
|
|
exprctx_t exprctx = { .symtab = &root_symtab };
|
|
parsectx_t parsectx = { &exprctx, ctx, properties };
|
|
auto rctx = ctx->render_context;
|
|
exprsym_t var_syms[] = {
|
|
{"output", &qfv_output_t_type, &sctx->output},
|
|
{"frames", &qfv_renderframeset_t_type, &rctx->frames},
|
|
{"msaaSamples", &VkSampleCountFlagBits_type, &ctx->msaaSamples},
|
|
{"physDevLimits", &VkPhysicalDeviceLimits_type,
|
|
&ctx->device->physDev->properties->limits },
|
|
{QFV_PROPERTIES, &cexpr_plitem, &parsectx.properties},
|
|
{}
|
|
};
|
|
exprtab_t vars_tab = { var_syms, 0 };
|
|
|
|
exprctx.external_variables = &vars_tab;
|
|
exprctx.messages = messages;
|
|
exprctx.hashctx = &sctx->hashctx;
|
|
exprctx.memsuper = memsuper;
|
|
|
|
cexpr_init_symtab (&vars_tab, &exprctx);
|
|
|
|
|
|
if (!parser (0, plist, object, messages, &parsectx)) {
|
|
for (int i = 0; i < PL_A_NumObjects (messages); i++) {
|
|
Sys_Printf ("%s\n", PL_String (PL_ObjectAtIndex (messages, i)));
|
|
}
|
|
return 0;
|
|
}
|
|
Hash_DelTable (vars_tab.tab);
|
|
PL_Release (messages);
|
|
|
|
return 1;
|
|
}
|
|
|
|
typedef struct {
|
|
uint32_t count;
|
|
VkImageCreateInfo *info;
|
|
} imagecreate_t;
|
|
|
|
typedef struct {
|
|
uint32_t count;
|
|
VkImageViewCreateInfo *info;
|
|
} imageviewcreate_t;
|
|
|
|
int
|
|
QFV_ParseOutput (vulkan_ctx_t *ctx, qfv_output_t *output, plitem_t *plist,
|
|
plitem_t *properties)
|
|
{
|
|
memsuper_t *memsuper = new_memsuper ();
|
|
int ret = 0;
|
|
qfv_output_t op = {};
|
|
|
|
if (parse_object (ctx, memsuper, plist, parse_qfv_output_t, &op,
|
|
properties)) {
|
|
memcpy (output, &op, sizeof (*output));
|
|
ret = 1;
|
|
}
|
|
delete_memsuper (memsuper);
|
|
return ret;
|
|
}
|
|
|
|
int vulkan_frame_count;
|
|
static cvar_t vulkan_frame_count_cvar = {
|
|
.name = "vulkan_frame_count",
|
|
.description =
|
|
"Number of frames to render in the background. More frames can "
|
|
"increase performance, but at the cost of latency. The default of 3 is"
|
|
" recommended.",
|
|
.default_value = "3",
|
|
.flags = CVAR_NONE,
|
|
.value = { .type = &cexpr_int, .value = &vulkan_frame_count },
|
|
};
|
|
int vulkan_presentation_mode;
|
|
static cvar_t vulkan_presentation_mode_cvar = {
|
|
.name = "vulkan_presentation_mode",
|
|
.description =
|
|
"desired presentation mode (may fall back to fifo).",
|
|
.default_value = "mailbox",
|
|
.flags = CVAR_NONE,
|
|
.value = {
|
|
.type = &VkPresentModeKHR_type,
|
|
.value = &vulkan_presentation_mode,
|
|
},
|
|
};
|
|
int msaaSamples;
|
|
static cvar_t msaaSamples_cvar = {
|
|
.name = "msaaSamples",
|
|
.description =
|
|
"desired MSAA sample size.",
|
|
.default_value = "VK_SAMPLE_COUNT_1_BIT",
|
|
.flags = CVAR_NONE,
|
|
.value = { .type = &VkSampleCountFlagBits_type, .value = &msaaSamples },
|
|
};
|
|
static exprenum_t validation_enum;
|
|
static exprtype_t validation_type = {
|
|
.name = "vulkan_use_validation",
|
|
.size = sizeof (int),
|
|
.binops = cexpr_flag_binops,
|
|
.unops = cexpr_flag_unops,
|
|
.data = &validation_enum,
|
|
.get_string = cexpr_flags_get_string,
|
|
};
|
|
|
|
static int validation_values[] = {
|
|
0,
|
|
VK_DEBUG_UTILS_MESSAGE_SEVERITY_FLAG_BITS_MAX_ENUM_EXT,
|
|
};
|
|
static exprsym_t validation_symbols[] = {
|
|
{"none", &validation_type, validation_values + 0},
|
|
{"all", &validation_type, validation_values + 1},
|
|
{}
|
|
};
|
|
static exprtab_t validation_symtab = {
|
|
validation_symbols,
|
|
};
|
|
static exprenum_t validation_enum = {
|
|
&validation_type,
|
|
&validation_symtab,
|
|
};
|
|
static cvar_t vulkan_use_validation_cvar = {
|
|
.name = "vulkan_use_validation",
|
|
.description =
|
|
"enable KRONOS Validation Layer if available (requires instance "
|
|
"restart).",
|
|
.default_value = "error|warning",
|
|
.flags = CVAR_NONE,
|
|
.value = { .type = &validation_type, .value = &vulkan_use_validation },
|
|
};
|
|
|
|
static void
|
|
vulkan_frame_count_f (void *data, const cvar_t *cvar)
|
|
{
|
|
if (vulkan_frame_count < 1) {
|
|
Sys_Printf ("Invalid frame count: %d. Setting to 1\n",
|
|
vulkan_frame_count);
|
|
vulkan_frame_count = 1;
|
|
}
|
|
}
|
|
|
|
void
|
|
Vulkan_Init_Cvars (void)
|
|
{
|
|
int num_syms = 0;
|
|
for (exprsym_t *sym = VkDebugUtilsMessageSeverityFlagBitsEXT_symbols;
|
|
sym->name; sym++, num_syms++) {
|
|
}
|
|
for (exprsym_t *sym = validation_symbols; sym->name; sym++, num_syms++) {
|
|
}
|
|
validation_symtab.symbols = calloc (num_syms + 1, sizeof (exprsym_t));
|
|
num_syms = 0;
|
|
for (exprsym_t *sym = VkDebugUtilsMessageSeverityFlagBitsEXT_symbols;
|
|
sym->name; sym++, num_syms++) {
|
|
validation_symtab.symbols[num_syms] = *sym;
|
|
validation_symtab.symbols[num_syms].type = &validation_type;
|
|
}
|
|
for (exprsym_t *sym = validation_symbols; sym->name; sym++, num_syms++) {
|
|
validation_symtab.symbols[num_syms] = *sym;
|
|
}
|
|
Cvar_Register (&vulkan_use_validation_cvar, 0, 0);
|
|
// FIXME implement fallback choices (instead of just fifo)
|
|
Cvar_Register (&vulkan_presentation_mode_cvar, 0, 0);
|
|
Cvar_Register (&vulkan_frame_count_cvar, vulkan_frame_count_f, 0);
|
|
Cvar_Register (&msaaSamples_cvar, 0, 0);
|
|
}
|
|
|
|
static exprsym_t builtin_plist_syms[] = {
|
|
{ .name = "main_def",
|
|
.value = (void *)
|
|
#include "libs/video/renderer/vulkan/rp_main_def.plc"
|
|
},
|
|
{ .name = "main_fwd",
|
|
.value = (void *)
|
|
#include "libs/video/renderer/vulkan/rp_main_fwd.plc"
|
|
},
|
|
{ .name = "smp_quake",
|
|
.value = (void *)
|
|
#include "libs/video/renderer/vulkan/smp_quake.plc"
|
|
},
|
|
{}
|
|
};
|
|
static plitem_t **builtin_plists;
|
|
static exprtab_t builtin_configs = { .symbols = builtin_plist_syms };
|
|
|
|
static void
|
|
build_configs (scriptctx_t *sctx)
|
|
{
|
|
int num_plists = 0;
|
|
for (exprsym_t *sym = builtin_plist_syms; sym->name; sym++) {
|
|
num_plists++;
|
|
}
|
|
builtin_plists = malloc (num_plists * sizeof (plitem_t *));
|
|
num_plists = 0;
|
|
for (exprsym_t *sym = builtin_plist_syms; sym->name; sym++) {
|
|
plitem_t *item = PL_GetDictionary (sym->value, &sctx->hashctx);
|
|
if (!item) {
|
|
// Syntax errors in the compiled-in plists are unrecoverable
|
|
Sys_Error ("Error parsing plist for %s", sym->name);
|
|
}
|
|
builtin_plists[num_plists] = item;
|
|
sym->value = &builtin_plists[num_plists];
|
|
sym->type = &cexpr_plitem;
|
|
num_plists++;
|
|
}
|
|
exprctx_t ectx = { .hashctx = &sctx->hashctx };
|
|
cexpr_init_symtab (&builtin_configs, &ectx);
|
|
}
|
|
|
|
static void
|
|
delete_configs (void)
|
|
{
|
|
int num_plists = 0;
|
|
for (exprsym_t *sym = builtin_plist_syms; sym->name; sym++) {
|
|
PL_Release (builtin_plists[num_plists]);
|
|
num_plists++;
|
|
}
|
|
free (builtin_plists);
|
|
}
|
|
|
|
plitem_t *
|
|
Vulkan_GetConfig (vulkan_ctx_t *ctx, const char *name)
|
|
{
|
|
scriptctx_t *sctx = ctx->script_context;
|
|
if (!builtin_configs.tab) {
|
|
build_configs (sctx);
|
|
}
|
|
|
|
plitem_t *config = 0;
|
|
exprval_t result = { .type = &cexpr_plitem, .value = &config };
|
|
exprctx_t ectx = {
|
|
.result = &result,
|
|
.symtab = &builtin_configs,
|
|
.memsuper = new_memsuper (),
|
|
.hashctx = &sctx->hashctx,
|
|
.messages = PL_NewArray (),
|
|
};
|
|
if (cexpr_eval_string (name, &ectx)) {
|
|
dstring_t *msg = dstring_newstr ();
|
|
|
|
for (int i = 0; i < PL_A_NumObjects (ectx.messages); i++) {
|
|
dasprintf (msg, "%s\n",
|
|
PL_String (PL_ObjectAtIndex (ectx.messages, i)));
|
|
}
|
|
Sys_Printf ("%s", msg->str);
|
|
dstring_delete (msg);
|
|
config = 0;
|
|
}
|
|
PL_Release (ectx.messages);
|
|
delete_memsuper (ectx.memsuper);
|
|
return config;
|
|
}
|
|
|
|
void Vulkan_Script_Init (vulkan_ctx_t *ctx)
|
|
{
|
|
scriptctx_t *sctx = calloc (1, sizeof (scriptctx_t));
|
|
sctx->vctx = ctx;
|
|
ctx->script_context = sctx;
|
|
|
|
exprctx_t ectx = {};
|
|
enum_symtab = Hash_NewTable (61, enum_symtab_getkey, 0, 0, &sctx->hashctx);
|
|
parser_table = Hash_NewTable (61, parser_getkey, 0, 0, &sctx->hashctx);
|
|
//FIXME using the script context's hashctx causes the cvar system to access
|
|
//freed hash links due to shutdown order issues. The proper fix would be to
|
|
//create a symtabs shutdown (for thread safety), but this works for now.
|
|
ectx.hashctx = 0;//&sctx->hashctx;
|
|
vkgen_init_symtabs (&ectx);
|
|
cexpr_init_symtab (&qfv_output_t_symtab, &ectx);
|
|
cexpr_init_symtab (&qfv_renderframeset_t_symtab, &ectx);
|
|
cexpr_init_symtab (&data_array_symtab, &ectx);
|
|
|
|
sctx->shaderModules = handlref_symtab (shaderModule_free, sctx);
|
|
sctx->setLayouts = handlref_symtab (setLayout_free, sctx);
|
|
sctx->pipelineLayouts = handlref_symtab (pipelineLayout_free, sctx);
|
|
sctx->descriptorPools = handlref_symtab (descriptorPool_free, sctx);
|
|
sctx->samplers = handlref_symtab (sampler_free, sctx);
|
|
sctx->images = handlref_symtab (image_free, sctx);
|
|
sctx->imageViews = handlref_symtab (imageView_free, sctx);
|
|
sctx->renderpasses = handlref_symtab (renderpass_free, sctx);
|
|
}
|
|
|
|
static void
|
|
clear_table (hashtab_t **table)
|
|
{
|
|
if (*table) {
|
|
hashtab_t *tab = *table;
|
|
*table = 0;
|
|
Hash_DelTable (tab);
|
|
}
|
|
}
|
|
|
|
void Vulkan_Script_Shutdown (vulkan_ctx_t *ctx)
|
|
{
|
|
scriptctx_t *sctx = ctx->script_context;
|
|
|
|
clear_table (&sctx->pipelineLayouts);
|
|
clear_table (&sctx->setLayouts);
|
|
clear_table (&sctx->shaderModules);
|
|
clear_table (&sctx->descriptorPools);
|
|
clear_table (&sctx->samplers);
|
|
|
|
delete_configs ();
|
|
|
|
Hash_DelContext (sctx->hashctx);
|
|
|
|
free (sctx);
|
|
}
|
|
|
|
void Vulkan_Script_SetOutput (vulkan_ctx_t *ctx, qfv_output_t *output)
|
|
{
|
|
scriptctx_t *sctx = ctx->script_context;
|
|
sctx->output = *output;
|
|
}
|
|
|
|
exprtab_t *
|
|
QFV_CreateSymtab (plitem_t *dict, const char *properties,
|
|
const char **extra_items, exprsym_t *extra_syms,
|
|
exprctx_t *ectx)
|
|
{
|
|
plitem_t *props = PL_ObjectForKey (dict, properties);
|
|
int num_keys = PL_D_NumKeys (props);
|
|
int num_extra = 0;
|
|
int num_syms = 0;
|
|
|
|
if (extra_items) {
|
|
for (const char **e = extra_items; *e; e++) {
|
|
if (PL_ObjectForKey (dict, *e)) {
|
|
num_extra++;
|
|
}
|
|
}
|
|
}
|
|
if (extra_syms) {
|
|
for (exprsym_t *sym = extra_syms; sym->name; sym++, num_syms++) { }
|
|
}
|
|
|
|
int total_items = num_keys + num_extra;
|
|
int total_syms = total_items + num_syms;
|
|
size_t size = (sizeof (exprtab_t)
|
|
+ (total_syms + 1) * sizeof (exprsym_t)
|
|
+ total_items * sizeof (plitem_t *));
|
|
exprtab_t *symtab = malloc (size);
|
|
*symtab = (exprtab_t) {
|
|
.symbols = (exprsym_t *) &symtab[1],
|
|
};
|
|
plitem_t **items = (plitem_t **) &symtab->symbols[total_syms + 1];
|
|
for (int i = 0; i < num_keys; i++) {
|
|
symtab->symbols[i] = (exprsym_t) {
|
|
.name = PL_KeyAtIndex (props, i),
|
|
.type = &cexpr_plitem,
|
|
.value = items + i,
|
|
};
|
|
items[i] = PL_ObjectForKey (props, symtab->symbols[i].name);
|
|
}
|
|
for (int i = 0, j = 0; num_extra && extra_items[i]; i++) {
|
|
plitem_t *val = PL_ObjectForKey (dict, extra_items[i]);
|
|
if (val) {
|
|
symtab->symbols[num_keys + j] = (exprsym_t) {
|
|
.name = extra_items[i],
|
|
.type = &cexpr_plitem,
|
|
.value = items + num_keys + j,
|
|
};
|
|
items[num_keys + j] = val;
|
|
j++;
|
|
}
|
|
}
|
|
for (int i = 0; i < num_syms; i++) {
|
|
symtab->symbols[total_items + i] = extra_syms[i];
|
|
}
|
|
symtab->symbols[total_syms] = (exprsym_t) { };
|
|
|
|
cexpr_init_symtab (symtab, ectx);
|
|
return symtab;
|
|
}
|
|
|
|
void
|
|
QFV_DestroySymtab (exprtab_t *tab)
|
|
{
|
|
Hash_DelTable (tab->tab);
|
|
free (tab);
|
|
}
|
|
|
|
struct qfv_jobinfo_s *
|
|
QFV_ParseJobInfo (vulkan_ctx_t *ctx, plitem_t *item, qfv_renderctx_t *rctx)
|
|
{
|
|
memsuper_t *memsuper = new_memsuper ();
|
|
qfv_jobinfo_t *ji = cmemalloc (memsuper, sizeof (qfv_jobinfo_t));
|
|
*ji = (qfv_jobinfo_t) { .memsuper = memsuper };
|
|
|
|
scriptctx_t *sctx = ctx->script_context;
|
|
plitem_t *messages = PL_NewArray ();
|
|
|
|
exprctx_t exprctx = {
|
|
.symtab = &root_symtab,
|
|
.messages = messages,
|
|
.hashctx = &sctx->hashctx,
|
|
.memsuper = memsuper,
|
|
};
|
|
parsectx_t parsectx = {
|
|
.ectx = &exprctx,
|
|
.vctx = ctx,
|
|
.data = rctx,
|
|
};
|
|
|
|
static const char *extra_items[] = {
|
|
"images",
|
|
"imageviews",
|
|
"renderpasses",
|
|
0
|
|
};
|
|
exprsym_t var_syms[] = {
|
|
{"render_output", &qfv_output_t_type, &sctx->output},
|
|
{"frames", &qfv_renderframeset_t_type, &rctx->frames},
|
|
{"msaaSamples", &VkSampleCountFlagBits_type, &ctx->msaaSamples},
|
|
{"physDevLimits", &VkPhysicalDeviceLimits_type,
|
|
&ctx->device->physDev->properties->limits },
|
|
{}
|
|
};
|
|
exprctx.external_variables = QFV_CreateSymtab (item, "properties",
|
|
extra_items, var_syms,
|
|
&exprctx);
|
|
|
|
int ret;
|
|
if (!(ret = parse_qfv_jobinfo_t (0, item, ji, messages, &parsectx))) {
|
|
for (int i = 0; i < PL_A_NumObjects (messages); i++) {
|
|
Sys_Printf ("%s\n", PL_String (PL_ObjectAtIndex (messages, i)));
|
|
}
|
|
}
|
|
QFV_DestroySymtab (exprctx.external_variables);
|
|
PL_Release (messages);
|
|
if (!ret) {
|
|
delete_memsuper (memsuper);
|
|
ji = 0;
|
|
}
|
|
|
|
return ji;
|
|
}
|
|
|
|
struct qfv_samplerinfo_s *
|
|
QFV_ParseSamplerInfo (vulkan_ctx_t *ctx, plitem_t *item, qfv_renderctx_t *rctx)
|
|
{
|
|
memsuper_t *memsuper = new_memsuper ();
|
|
qfv_samplerinfo_t *si = cmemalloc (memsuper, sizeof (qfv_samplerinfo_t));
|
|
*si = (qfv_samplerinfo_t) { .memsuper = memsuper };
|
|
|
|
scriptctx_t *sctx = ctx->script_context;
|
|
plitem_t *messages = PL_NewArray ();
|
|
|
|
exprctx_t exprctx = {
|
|
.symtab = &root_symtab,
|
|
.messages = messages,
|
|
.hashctx = &sctx->hashctx,
|
|
.memsuper = memsuper,
|
|
};
|
|
parsectx_t parsectx = {
|
|
.ectx = &exprctx,
|
|
.vctx = ctx,
|
|
.data = rctx,
|
|
};
|
|
|
|
exprsym_t var_syms[] = {
|
|
{"physDevLimits", &VkPhysicalDeviceLimits_type,
|
|
&ctx->device->physDev->properties->limits },
|
|
{}
|
|
};
|
|
exprctx.external_variables = QFV_CreateSymtab (item, "properties",
|
|
0, var_syms, &exprctx);
|
|
|
|
int ret;
|
|
if (!(ret = parse_qfv_samplerinfo_t (0, item, si, messages, &parsectx))) {
|
|
for (int i = 0; i < PL_A_NumObjects (messages); i++) {
|
|
Sys_Printf ("%s\n", PL_String (PL_ObjectAtIndex (messages, i)));
|
|
}
|
|
}
|
|
QFV_DestroySymtab (exprctx.external_variables);
|
|
PL_Release (messages);
|
|
if (!ret) {
|
|
delete_memsuper (memsuper);
|
|
si = 0;
|
|
}
|
|
|
|
return si;
|
|
}
|
|
|
|
int
|
|
QFV_ParseLayoutInfo (vulkan_ctx_t *ctx, memsuper_t *memsuper,
|
|
exprtab_t *symtab, const char *ref,
|
|
qfv_layoutinfo_t *layout)
|
|
{
|
|
*layout = (qfv_layoutinfo_t) {};
|
|
|
|
scriptctx_t *sctx = ctx->script_context;
|
|
plitem_t *messages = PL_NewArray ();
|
|
|
|
exprctx_t exprctx = {
|
|
.symtab = &root_symtab,
|
|
.messages = messages,
|
|
.hashctx = &sctx->hashctx,
|
|
.memsuper = memsuper,
|
|
.external_variables = symtab,
|
|
};
|
|
parsectx_t parsectx = {
|
|
.ectx = &exprctx,
|
|
.vctx = ctx,
|
|
};
|
|
|
|
plitem_t *item = PL_NewString (ref);
|
|
int ret;
|
|
if (!(ret = parse_qfv_layoutinfo_t (0, item, layout, messages,
|
|
&parsectx))) {
|
|
for (int i = 0; i < PL_A_NumObjects (messages); i++) {
|
|
Sys_Printf ("%s\n", PL_String (PL_ObjectAtIndex (messages, i)));
|
|
}
|
|
}
|
|
PL_Release (messages);
|
|
PL_Release (item);
|
|
|
|
return ret;
|
|
}
|