blob: 6246347972ee4d1a44b7b5c5d2cb08262d1fb537 [file] [log] [blame]
/*
* Copyright 2019 Valve Corporation
*
* SPDX-License-Identifier: MIT
*/
#include "ac_shader_args.h"
#include "nir/nir_defines.h"
#include <assert.h>
void ac_add_arg(struct ac_shader_args *info, enum ac_arg_regfile regfile, unsigned size,
enum ac_arg_type type, struct ac_arg *arg)
{
assert(info->arg_count < AC_MAX_ARGS);
assert(nir_num_components_valid(size));
unsigned offset;
if (regfile == AC_ARG_SGPR) {
offset = info->num_sgprs_used;
info->num_sgprs_used += size;
} else {
assert(regfile == AC_ARG_VGPR);
offset = info->num_vgprs_used;
info->num_vgprs_used += size;
}
info->args[info->arg_count].file = regfile;
info->args[info->arg_count].offset = offset;
info->args[info->arg_count].size = size;
info->args[info->arg_count].type = type;
if (arg) {
arg->arg_index = info->arg_count;
arg->used = true;
}
info->arg_count++;
}
void ac_add_return(struct ac_shader_args *info, enum ac_arg_regfile regfile)
{
assert(info->return_count < AC_MAX_ARGS);
if (regfile == AC_ARG_SGPR) {
/* SGPRs must be inserted before VGPRs. */
assert(info->num_vgprs_returned == 0);
info->num_sgprs_returned++;;
} else {
assert(regfile == AC_ARG_VGPR);
info->num_vgprs_returned++;
}
info->return_count++;
}
void ac_add_preserved(struct ac_shader_args *info, const struct ac_arg *arg)
{
info->args[arg->arg_index].preserved = true;
}
void ac_compact_ps_vgpr_args(struct ac_shader_args *info, uint32_t spi_ps_input)
{
/* LLVM optimizes away unused FS inputs and computes spi_ps_input_addr itself and then
* communicates the results back via the ELF binary. Mirror what LLVM does by re-mapping the
* VGPR arguments here.
*/
unsigned vgpr_arg = 0;
unsigned vgpr_reg = 0;
for (unsigned i = 0; i < info->arg_count; i++) {
if (info->args[i].file != AC_ARG_VGPR)
continue;
if (!(spi_ps_input & (1 << vgpr_arg))) {
info->args[i].skip = true;
} else {
info->args[i].offset = vgpr_reg;
vgpr_reg += info->args[i].size;
}
vgpr_arg++;
}
info->num_vgprs_used = vgpr_reg;
}