2010-03-28 21:16:09 +02:00
|
|
|
/*
|
|
|
|
* Direct3D shader assembler
|
|
|
|
*
|
|
|
|
* Copyright 2008 Stefan Dösinger
|
|
|
|
* Copyright 2009 Matteo Bruni
|
|
|
|
*
|
|
|
|
* This library 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.
|
|
|
|
*
|
|
|
|
* This library 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 this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
%{
|
|
|
|
#include "config.h"
|
|
|
|
#include "wine/port.h"
|
|
|
|
#include "wine/debug.h"
|
|
|
|
|
|
|
|
#include "d3dx9_36_private.h"
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(asmshader);
|
|
|
|
|
|
|
|
struct asm_parser asm_ctx;
|
|
|
|
|
|
|
|
/* Needed lexer functions declarations */
|
|
|
|
void asmshader_error(const char *s);
|
|
|
|
int asmshader_lex(void);
|
|
|
|
|
|
|
|
void set_rel_reg(struct shader_reg *reg, struct rel_reg *rel) {
|
2010-05-05 19:55:26 +02:00
|
|
|
/* We can have an additional offset without true relative addressing
|
|
|
|
* ex. c2[ 4 ] */
|
|
|
|
reg->regnum += rel->additional_offset;
|
|
|
|
if(!rel->has_rel_reg) {
|
|
|
|
reg->rel_reg = NULL;
|
|
|
|
} else {
|
|
|
|
reg->rel_reg = asm_alloc(sizeof(*reg->rel_reg));
|
|
|
|
if(!reg->rel_reg) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
reg->rel_reg->type = rel->type;
|
|
|
|
reg->rel_reg->swizzle = rel->swizzle;
|
|
|
|
reg->rel_reg->regnum = rel->rel_regnum;
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
%}
|
|
|
|
|
|
|
|
%union {
|
2010-05-05 19:55:26 +02:00
|
|
|
struct {
|
|
|
|
float val;
|
|
|
|
BOOL integer;
|
|
|
|
} immval;
|
2010-03-28 21:16:09 +02:00
|
|
|
unsigned int regnum;
|
|
|
|
struct shader_reg reg;
|
2010-04-21 16:15:46 +02:00
|
|
|
DWORD srcmod;
|
2010-04-21 16:15:46 +02:00
|
|
|
DWORD writemask;
|
2010-03-28 21:16:09 +02:00
|
|
|
struct {
|
|
|
|
DWORD writemask;
|
2010-04-21 16:15:46 +02:00
|
|
|
DWORD idx;
|
|
|
|
DWORD last;
|
|
|
|
} wm_components;
|
2010-03-28 21:16:09 +02:00
|
|
|
DWORD swizzle;
|
2010-04-21 16:15:46 +02:00
|
|
|
struct {
|
|
|
|
DWORD swizzle;
|
|
|
|
DWORD idx;
|
|
|
|
} sw_components;
|
|
|
|
DWORD component;
|
2010-03-28 21:16:09 +02:00
|
|
|
struct {
|
|
|
|
DWORD mod;
|
|
|
|
DWORD shift;
|
|
|
|
} modshift;
|
|
|
|
struct rel_reg rel_reg;
|
|
|
|
struct src_regs sregs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Common instructions between vertex and pixel shaders */
|
2010-05-05 20:21:10 +02:00
|
|
|
%token INSTR_ADD
|
|
|
|
%token INSTR_NOP
|
2010-03-28 21:16:09 +02:00
|
|
|
%token INSTR_MOV
|
2010-05-05 20:21:10 +02:00
|
|
|
%token INSTR_SUB
|
|
|
|
%token INSTR_MAD
|
|
|
|
%token INSTR_MUL
|
|
|
|
%token INSTR_RCP
|
|
|
|
%token INSTR_RSQ
|
|
|
|
%token INSTR_DP3
|
|
|
|
%token INSTR_DP4
|
|
|
|
%token INSTR_MIN
|
|
|
|
%token INSTR_MAX
|
|
|
|
%token INSTR_SLT
|
|
|
|
%token INSTR_SGE
|
|
|
|
%token INSTR_ABS
|
|
|
|
%token INSTR_EXP
|
|
|
|
%token INSTR_LOG
|
|
|
|
%token INSTR_EXPP
|
|
|
|
%token INSTR_LOGP
|
|
|
|
%token INSTR_DST
|
|
|
|
%token INSTR_LRP
|
|
|
|
%token INSTR_FRC
|
|
|
|
%token INSTR_POW
|
|
|
|
%token INSTR_CRS
|
|
|
|
%token INSTR_SGN
|
|
|
|
%token INSTR_NRM
|
|
|
|
%token INSTR_SINCOS
|
|
|
|
%token INSTR_M4x4
|
|
|
|
%token INSTR_M4x3
|
|
|
|
%token INSTR_M3x4
|
|
|
|
%token INSTR_M3x3
|
|
|
|
%token INSTR_M3x2
|
|
|
|
%token INSTR_TEXLDL
|
|
|
|
|
|
|
|
/* Vertex shader only instructions */
|
|
|
|
%token INSTR_LIT
|
|
|
|
%token INSTR_MOVA
|
2010-03-28 21:16:09 +02:00
|
|
|
|
|
|
|
/* Registers */
|
|
|
|
%token <regnum> REG_TEMP
|
2010-05-05 16:56:09 +02:00
|
|
|
%token <regnum> REG_OUTPUT
|
|
|
|
%token <regnum> REG_INPUT
|
2010-03-28 21:16:09 +02:00
|
|
|
%token <regnum> REG_CONSTFLOAT
|
2010-05-05 16:56:09 +02:00
|
|
|
%token <regnum> REG_CONSTINT
|
|
|
|
%token <regnum> REG_CONSTBOOL
|
|
|
|
%token <regnum> REG_TEXTURE
|
|
|
|
%token <regnum> REG_SAMPLER
|
|
|
|
%token <regnum> REG_TEXCRDOUT
|
|
|
|
%token REG_OPOS
|
|
|
|
%token REG_OFOG
|
|
|
|
%token REG_OPTS
|
|
|
|
%token <regnum> REG_VERTEXCOLOR
|
|
|
|
%token <regnum> REG_FRAGCOLOR
|
|
|
|
%token REG_FRAGDEPTH
|
|
|
|
%token REG_VPOS
|
|
|
|
%token REG_VFACE
|
|
|
|
%token REG_ADDRESS
|
|
|
|
%token REG_LOOP
|
|
|
|
%token REG_PREDICATE
|
|
|
|
%token <regnum> REG_LABEL
|
2010-03-28 21:16:09 +02:00
|
|
|
|
|
|
|
/* Version tokens */
|
|
|
|
%token VER_VS10
|
|
|
|
%token VER_VS11
|
|
|
|
%token VER_VS20
|
|
|
|
%token VER_VS2X
|
|
|
|
%token VER_VS30
|
|
|
|
|
|
|
|
%token VER_PS10
|
|
|
|
%token VER_PS11
|
|
|
|
%token VER_PS12
|
|
|
|
%token VER_PS13
|
|
|
|
%token VER_PS14
|
|
|
|
%token VER_PS20
|
|
|
|
%token VER_PS2X
|
|
|
|
%token VER_PS30
|
|
|
|
|
2010-04-21 16:15:46 +02:00
|
|
|
/* Output modifiers */
|
|
|
|
%token MOD_SAT
|
|
|
|
%token MOD_PP
|
|
|
|
%token MOD_CENTROID
|
|
|
|
|
2010-04-21 16:15:46 +02:00
|
|
|
/* Source register modifiers */
|
|
|
|
%token SMOD_ABS
|
|
|
|
|
2010-04-21 16:15:46 +02:00
|
|
|
/* Misc stuff */
|
|
|
|
%token <component> COMPONENT
|
2010-05-05 19:55:26 +02:00
|
|
|
%token <immval> IMMVAL
|
2010-03-28 21:16:09 +02:00
|
|
|
|
|
|
|
%type <reg> dreg_name
|
|
|
|
%type <reg> dreg
|
|
|
|
%type <reg> sreg_name
|
2010-05-05 19:55:26 +02:00
|
|
|
%type <reg> relreg_name
|
2010-03-28 21:16:09 +02:00
|
|
|
%type <reg> sreg
|
2010-04-21 16:15:46 +02:00
|
|
|
%type <srcmod> smod
|
2010-04-21 16:15:46 +02:00
|
|
|
%type <writemask> writemask
|
|
|
|
%type <wm_components> wm_components
|
2010-03-28 21:16:09 +02:00
|
|
|
%type <swizzle> swizzle
|
2010-04-21 16:15:46 +02:00
|
|
|
%type <sw_components> sw_components
|
2010-03-28 21:16:09 +02:00
|
|
|
%type <modshift> omods
|
2010-04-21 16:15:46 +02:00
|
|
|
%type <modshift> omodifier
|
2010-03-28 21:16:09 +02:00
|
|
|
%type <rel_reg> rel_reg
|
2010-05-05 19:55:26 +02:00
|
|
|
%type <immval> immsum
|
2010-03-28 21:16:09 +02:00
|
|
|
%type <sregs> sregs
|
|
|
|
|
|
|
|
%%
|
|
|
|
|
|
|
|
shader: version_marker instructions
|
|
|
|
{
|
|
|
|
asm_ctx.funcs->end(&asm_ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
version_marker: VER_VS10
|
|
|
|
{
|
|
|
|
TRACE("Vertex shader 1.0\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_VS11
|
|
|
|
{
|
|
|
|
TRACE("Vertex shader 1.1\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_VS20
|
|
|
|
{
|
|
|
|
TRACE("Vertex shader 2.0\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_VS2X
|
|
|
|
{
|
|
|
|
TRACE("Vertex shader 2.x\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_VS30
|
|
|
|
{
|
|
|
|
TRACE("Vertex shader 3.0\n");
|
|
|
|
create_vs30_parser(&asm_ctx);
|
|
|
|
}
|
|
|
|
| VER_PS10
|
|
|
|
{
|
|
|
|
TRACE("Pixel shader 1.0\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_PS11
|
|
|
|
{
|
|
|
|
TRACE("Pixel shader 1.1\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_PS12
|
|
|
|
{
|
|
|
|
TRACE("Pixel shader 1.2\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_PS13
|
|
|
|
{
|
|
|
|
TRACE("Pixel shader 1.3\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_PS14
|
|
|
|
{
|
|
|
|
TRACE("Pixel shader 1.4\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_PS20
|
|
|
|
{
|
|
|
|
TRACE("Pixel shader 2.0\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_PS2X
|
|
|
|
{
|
|
|
|
TRACE("Pixel shader 2.x\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
| VER_PS30
|
|
|
|
{
|
|
|
|
TRACE("Pixel shader 3.0\n");
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
YYABORT;
|
|
|
|
}
|
|
|
|
|
|
|
|
instructions: /* empty */
|
|
|
|
| instructions complexinstr
|
|
|
|
{
|
|
|
|
/* Nothing to do */
|
|
|
|
}
|
|
|
|
|
|
|
|
complexinstr: instruction
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2010-05-05 20:21:10 +02:00
|
|
|
instruction: INSTR_ADD omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("ADD\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_ADD, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_NOP
|
|
|
|
{
|
|
|
|
TRACE("NOP\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_NOP, 0, 0, 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
| INSTR_MOV omods dreg ',' sregs
|
2010-03-28 21:16:09 +02:00
|
|
|
{
|
|
|
|
TRACE("MOV\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_MOV, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
2010-05-05 20:21:10 +02:00
|
|
|
| INSTR_SUB omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("SUB\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_SUB, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_MAD omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("MAD\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_MAD, $2.mod, $2.shift, 0, &$3, &$5, 3);
|
|
|
|
}
|
|
|
|
| INSTR_MUL omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("MUL\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_MUL, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_RCP omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("RCP\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_RCP, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_RSQ omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("RSQ\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_RSQ, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_DP3 omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("DP3\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_DP3, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_DP4 omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("DP4\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_DP4, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_MIN omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("MIN\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_MIN, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_MAX omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("MAX\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_MAX, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_SLT omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("SLT\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_SLT, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_SGE omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("SGE\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_SGE, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_ABS omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("ABS\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_ABS, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_EXP omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("EXP\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_EXP, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_LOG omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("LOG\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_LOG, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_LOGP omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("LOGP\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_LOGP, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_EXPP omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("EXPP\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_EXPP, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_DST omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("DST\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_DST, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_LRP omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("LRP\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_LRP, $2.mod, $2.shift, 0, &$3, &$5, 3);
|
|
|
|
}
|
|
|
|
| INSTR_FRC omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("FRC\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_FRC, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_POW omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("POW\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_POW, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_CRS omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("CRS\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_CRS, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_SGN omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("SGN\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_SGN, $2.mod, $2.shift, 0, &$3, &$5, 3);
|
|
|
|
}
|
|
|
|
| INSTR_NRM omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("NRM\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_NRM, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_SINCOS omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("SINCOS\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_SINCOS, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_M4x4 omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("M4x4\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_M4x4, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_M4x3 omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("M4x3\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_M4x3, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_M3x4 omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("M3x4\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_M3x4, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_M3x3 omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("M3x3\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_M3x3, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_M3x2 omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("M3x2\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_M3x2, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_TEXLDL omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("TEXLDL\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_TEXLDL, $2.mod, $2.shift, 0, &$3, &$5, 2);
|
|
|
|
}
|
|
|
|
| INSTR_LIT omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("LIT\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_LIT, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
|
|
|
| INSTR_MOVA omods dreg ',' sregs
|
|
|
|
{
|
|
|
|
TRACE("MOVA\n");
|
|
|
|
asm_ctx.funcs->instr(&asm_ctx, BWRITERSIO_MOVA, $2.mod, $2.shift, 0, &$3, &$5, 1);
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
|
|
|
|
dreg: dreg_name rel_reg
|
|
|
|
{
|
|
|
|
$$.regnum = $1.regnum;
|
|
|
|
$$.type = $1.type;
|
|
|
|
$$.writemask = BWRITERSP_WRITEMASK_ALL;
|
|
|
|
$$.srcmod = BWRITERSPSM_NONE;
|
|
|
|
set_rel_reg(&$$, &$2);
|
|
|
|
}
|
2010-04-21 16:15:46 +02:00
|
|
|
| dreg_name writemask
|
|
|
|
{
|
|
|
|
$$.regnum = $1.regnum;
|
|
|
|
$$.type = $1.type;
|
|
|
|
$$.writemask = $2;
|
|
|
|
$$.srcmod = BWRITERSPSM_NONE;
|
|
|
|
$$.rel_reg = NULL;
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
|
|
|
|
dreg_name: REG_TEMP
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_TEMP;
|
|
|
|
}
|
2010-05-05 16:56:09 +02:00
|
|
|
| REG_OUTPUT
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_OUTPUT;
|
|
|
|
}
|
|
|
|
| REG_INPUT
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register v%u is not a valid destination register\n",
|
|
|
|
asm_ctx.line_no, $1);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_CONSTFLOAT
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register c%u is not a valid destination register\n",
|
|
|
|
asm_ctx.line_no, $1);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_CONSTINT
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register i%u is not a valid destination register\n",
|
|
|
|
asm_ctx.line_no, $1);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_CONSTBOOL
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register b%u is not a valid destination register\n",
|
|
|
|
asm_ctx.line_no, $1);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_TEXTURE
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_TEXTURE;
|
|
|
|
}
|
|
|
|
| REG_TEXCRDOUT
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_TEXCRDOUT;
|
|
|
|
}
|
|
|
|
| REG_SAMPLER
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register s%u is not a valid destination register\n",
|
|
|
|
asm_ctx.line_no, $1);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_OPOS
|
|
|
|
{
|
|
|
|
$$.regnum = BWRITERSRO_POSITION; $$.type = BWRITERSPR_RASTOUT;
|
|
|
|
}
|
|
|
|
| REG_OPTS
|
|
|
|
{
|
|
|
|
$$.regnum = BWRITERSRO_POINT_SIZE; $$.type = BWRITERSPR_RASTOUT;
|
|
|
|
}
|
|
|
|
| REG_OFOG
|
|
|
|
{
|
|
|
|
$$.regnum = BWRITERSRO_FOG; $$.type = BWRITERSPR_RASTOUT;
|
|
|
|
}
|
|
|
|
| REG_VERTEXCOLOR
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_ATTROUT;
|
|
|
|
}
|
|
|
|
| REG_FRAGCOLOR
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_COLOROUT;
|
|
|
|
}
|
|
|
|
| REG_FRAGDEPTH
|
|
|
|
{
|
|
|
|
$$.regnum = 0; $$.type = BWRITERSPR_DEPTHOUT;
|
|
|
|
}
|
|
|
|
| REG_PREDICATE
|
|
|
|
{
|
|
|
|
$$.regnum = 0; $$.type = BWRITERSPR_PREDICATE;
|
|
|
|
}
|
|
|
|
| REG_VPOS
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register vPos is not a valid destination register\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_VFACE
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register vFace is not a valid destination register\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_ADDRESS
|
|
|
|
{
|
|
|
|
/* index 0 is hardcoded for the addr register */
|
|
|
|
$$.regnum = 0; $$.type = BWRITERSPR_ADDR;
|
|
|
|
}
|
|
|
|
| REG_LOOP
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register aL is not a valid destination register\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
|
2010-04-21 16:15:46 +02:00
|
|
|
writemask: '.' wm_components
|
|
|
|
{
|
|
|
|
if($2.writemask == SWIZZLE_ERR) {
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Invalid writemask specified\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
/* Provide a correct writemask to prevent following complaints */
|
|
|
|
$$ = BWRITERSP_WRITEMASK_ALL;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$$ = $2.writemask;
|
|
|
|
TRACE("Writemask: %x\n", $$);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
wm_components: COMPONENT
|
|
|
|
{
|
|
|
|
$$.writemask = 1 << $1;
|
|
|
|
$$.last = $1;
|
|
|
|
$$.idx = 1;
|
|
|
|
}
|
|
|
|
| wm_components COMPONENT
|
|
|
|
{
|
|
|
|
if($1.writemask == SWIZZLE_ERR || $1.idx == 4)
|
|
|
|
/* Wrong writemask */
|
|
|
|
$$.writemask = SWIZZLE_ERR;
|
|
|
|
else {
|
|
|
|
if($2 <= $1.last)
|
|
|
|
$$.writemask = SWIZZLE_ERR;
|
|
|
|
else {
|
|
|
|
$$.writemask = $1.writemask | (1 << $2);
|
|
|
|
$$.idx = $1.idx + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-03-28 21:16:09 +02:00
|
|
|
swizzle: /* empty */
|
|
|
|
{
|
|
|
|
$$ = BWRITERVS_NOSWIZZLE;
|
|
|
|
TRACE("Default swizzle: %08x\n", $$);
|
|
|
|
}
|
2010-04-21 16:15:46 +02:00
|
|
|
| '.' sw_components
|
|
|
|
{
|
|
|
|
if($2.swizzle == SWIZZLE_ERR) {
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Invalid swizzle\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
/* Provide a correct swizzle to prevent following complaints */
|
|
|
|
$$ = BWRITERVS_NOSWIZZLE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
DWORD last, i;
|
|
|
|
|
|
|
|
$$ = $2.swizzle << BWRITERVS_SWIZZLE_SHIFT;
|
|
|
|
/* Fill the swizzle by extending the last component */
|
|
|
|
last = ($2.swizzle >> 2 * ($2.idx - 1)) & 0x03;
|
|
|
|
for(i = $2.idx; i < 4; i++){
|
|
|
|
$$ |= last << (BWRITERVS_SWIZZLE_SHIFT + 2 * i);
|
|
|
|
}
|
|
|
|
TRACE("Got a swizzle: %08x\n", $$);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
sw_components: COMPONENT
|
|
|
|
{
|
|
|
|
$$.swizzle = $1;
|
|
|
|
$$.idx = 1;
|
|
|
|
}
|
|
|
|
| sw_components COMPONENT
|
|
|
|
{
|
|
|
|
if($1.idx == 4) {
|
|
|
|
/* Too many sw_components */
|
|
|
|
$$.swizzle = SWIZZLE_ERR;
|
|
|
|
$$.idx = 4;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$$.swizzle = $1.swizzle | ($2 << 2 * $1.idx);
|
|
|
|
$$.idx = $1.idx + 1;
|
|
|
|
}
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
|
|
|
|
omods: /* Empty */
|
|
|
|
{
|
|
|
|
$$.mod = 0;
|
|
|
|
$$.shift = 0;
|
|
|
|
}
|
2010-04-21 16:15:46 +02:00
|
|
|
| omods omodifier
|
|
|
|
{
|
|
|
|
$$.mod = $1.mod | $2.mod;
|
|
|
|
if($1.shift && $2.shift) {
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: More than one shift flag\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
$$.shift = $1.shift;
|
|
|
|
} else {
|
|
|
|
$$.shift = $1.shift | $2.shift;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
omodifier: MOD_SAT
|
|
|
|
{
|
|
|
|
$$.mod = BWRITERSPDM_SATURATE;
|
|
|
|
$$.shift = 0;
|
|
|
|
}
|
|
|
|
| MOD_PP
|
|
|
|
{
|
|
|
|
$$.mod = BWRITERSPDM_PARTIALPRECISION;
|
|
|
|
$$.shift = 0;
|
|
|
|
}
|
|
|
|
| MOD_CENTROID
|
|
|
|
{
|
|
|
|
$$.mod = BWRITERSPDM_MSAMPCENTROID;
|
|
|
|
$$.shift = 0;
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
|
|
|
|
sregs: sreg
|
|
|
|
{
|
|
|
|
$$.reg[0] = $1;
|
|
|
|
$$.count = 1;
|
|
|
|
}
|
|
|
|
| sregs ',' sreg
|
|
|
|
{
|
|
|
|
if($$.count == MAX_SRC_REGS){
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Too many source registers in this instruction\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
$$.reg[$$.count++] = $3;
|
|
|
|
}
|
|
|
|
|
|
|
|
sreg: sreg_name rel_reg swizzle
|
|
|
|
{
|
|
|
|
$$.type = $1.type;
|
|
|
|
$$.regnum = $1.regnum;
|
|
|
|
$$.swizzle = $3;
|
|
|
|
$$.srcmod = BWRITERSPSM_NONE;
|
|
|
|
set_rel_reg(&$$, &$2);
|
|
|
|
}
|
2010-04-21 16:15:46 +02:00
|
|
|
| sreg_name rel_reg smod swizzle
|
|
|
|
{
|
|
|
|
$$.type = $1.type;
|
|
|
|
$$.regnum = $1.regnum;
|
|
|
|
set_rel_reg(&$$, &$2);
|
|
|
|
$$.srcmod = $3;
|
|
|
|
$$.swizzle = $4;
|
|
|
|
}
|
|
|
|
| '-' sreg_name rel_reg swizzle
|
|
|
|
{
|
|
|
|
$$.type = $2.type;
|
|
|
|
$$.regnum = $2.regnum;
|
|
|
|
$$.srcmod = BWRITERSPSM_NEG;
|
|
|
|
set_rel_reg(&$$, &$3);
|
|
|
|
$$.swizzle = $4;
|
|
|
|
}
|
|
|
|
| '-' sreg_name rel_reg smod swizzle
|
|
|
|
{
|
|
|
|
$$.type = $2.type;
|
|
|
|
$$.regnum = $2.regnum;
|
|
|
|
set_rel_reg(&$$, &$3);
|
|
|
|
switch($4) {
|
|
|
|
case BWRITERSPSM_ABS: $$.srcmod = BWRITERSPSM_ABSNEG; break;
|
|
|
|
default:
|
|
|
|
FIXME("Unhandled combination of NEGATE and %u\n", $4);
|
|
|
|
}
|
|
|
|
$$.swizzle = $5;
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
|
|
|
|
rel_reg: /* empty */
|
|
|
|
{
|
|
|
|
$$.has_rel_reg = FALSE;
|
|
|
|
$$.additional_offset = 0;
|
|
|
|
}
|
2010-05-05 19:55:26 +02:00
|
|
|
| '[' immsum ']'
|
|
|
|
{
|
|
|
|
$$.has_rel_reg = FALSE;
|
|
|
|
$$.additional_offset = $2.val;
|
|
|
|
}
|
|
|
|
| '[' relreg_name swizzle ']'
|
|
|
|
{
|
|
|
|
$$.has_rel_reg = TRUE;
|
|
|
|
$$.type = $2.type;
|
|
|
|
$$.additional_offset = 0;
|
|
|
|
$$.rel_regnum = $2.regnum;
|
|
|
|
$$.swizzle = $3;
|
|
|
|
}
|
|
|
|
| '[' immsum '+' relreg_name swizzle ']'
|
|
|
|
{
|
|
|
|
$$.has_rel_reg = TRUE;
|
|
|
|
$$.type = $4.type;
|
|
|
|
$$.additional_offset = $2.val;
|
|
|
|
$$.rel_regnum = $4.regnum;
|
|
|
|
$$.swizzle = $5;
|
|
|
|
}
|
|
|
|
| '[' relreg_name swizzle '+' immsum ']'
|
|
|
|
{
|
|
|
|
$$.has_rel_reg = TRUE;
|
|
|
|
$$.type = $2.type;
|
|
|
|
$$.additional_offset = $5.val;
|
|
|
|
$$.rel_regnum = $2.regnum;
|
|
|
|
$$.swizzle = $3;
|
|
|
|
}
|
|
|
|
| '[' immsum '+' relreg_name swizzle '+' immsum ']'
|
|
|
|
{
|
|
|
|
$$.has_rel_reg = TRUE;
|
|
|
|
$$.type = $4.type;
|
|
|
|
$$.additional_offset = $2.val + $7.val;
|
|
|
|
$$.rel_regnum = $4.regnum;
|
|
|
|
$$.swizzle = $5;
|
|
|
|
}
|
|
|
|
|
|
|
|
immsum: IMMVAL
|
|
|
|
{
|
|
|
|
if(!$1.integer) {
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Unexpected float %f\n",
|
|
|
|
asm_ctx.line_no, $1.val);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
}
|
|
|
|
$$.val = $1.val;
|
|
|
|
}
|
|
|
|
| immsum '+' IMMVAL
|
|
|
|
{
|
|
|
|
if(!$3.integer) {
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Unexpected float %f\n",
|
|
|
|
asm_ctx.line_no, $3.val);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
}
|
|
|
|
$$.val = $1.val + $3.val;
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
|
2010-04-21 16:15:46 +02:00
|
|
|
smod: SMOD_ABS
|
|
|
|
{
|
|
|
|
$$ = BWRITERSPSM_ABS;
|
|
|
|
}
|
|
|
|
|
2010-05-05 19:55:26 +02:00
|
|
|
relreg_name: REG_ADDRESS
|
|
|
|
{
|
|
|
|
$$.regnum = 0; $$.type = BWRITERSPR_ADDR;
|
|
|
|
}
|
|
|
|
| REG_LOOP
|
|
|
|
{
|
|
|
|
$$.regnum = 0; $$.type = BWRITERSPR_LOOP;
|
|
|
|
}
|
|
|
|
|
2010-03-28 21:16:09 +02:00
|
|
|
sreg_name: REG_TEMP
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_TEMP;
|
|
|
|
}
|
2010-05-05 16:56:09 +02:00
|
|
|
| REG_OUTPUT
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register o%u is not a valid source register\n",
|
|
|
|
asm_ctx.line_no, $1);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_INPUT
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_INPUT;
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
| REG_CONSTFLOAT
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_CONST;
|
|
|
|
}
|
2010-05-05 16:56:09 +02:00
|
|
|
| REG_CONSTINT
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_CONSTINT;
|
|
|
|
}
|
|
|
|
| REG_CONSTBOOL
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_CONSTBOOL;
|
|
|
|
}
|
|
|
|
| REG_TEXTURE
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_TEXTURE;
|
|
|
|
}
|
|
|
|
| REG_TEXCRDOUT
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register oT%u is not a valid source register\n",
|
|
|
|
asm_ctx.line_no, $1);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_SAMPLER
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_SAMPLER;
|
|
|
|
}
|
|
|
|
| REG_OPOS
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register oPos is not a valid source register\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_OFOG
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register oFog is not a valid source register\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_VERTEXCOLOR
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register oD%u is not a valid source register\n",
|
|
|
|
asm_ctx.line_no, $1);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_FRAGCOLOR
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register oC%u is not a valid source register\n",
|
|
|
|
asm_ctx.line_no, $1);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_FRAGDEPTH
|
|
|
|
{
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Register oDepth is not a valid source register\n",
|
|
|
|
asm_ctx.line_no);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_WARN);
|
|
|
|
}
|
|
|
|
| REG_PREDICATE
|
|
|
|
{
|
|
|
|
$$.regnum = 0; $$.type = BWRITERSPR_PREDICATE;
|
|
|
|
}
|
|
|
|
| REG_VPOS
|
|
|
|
{
|
|
|
|
$$.regnum = 0; $$.type = BWRITERSPR_MISCTYPE;
|
|
|
|
}
|
|
|
|
| REG_VFACE
|
|
|
|
{
|
|
|
|
$$.regnum = 1; $$.type = BWRITERSPR_MISCTYPE;
|
|
|
|
}
|
|
|
|
| REG_ADDRESS
|
|
|
|
{
|
|
|
|
$$.regnum = 0; $$.type = BWRITERSPR_ADDR;
|
|
|
|
}
|
|
|
|
| REG_LOOP
|
|
|
|
{
|
|
|
|
$$.regnum = 0; $$.type = BWRITERSPR_LOOP;
|
|
|
|
}
|
|
|
|
| REG_LABEL
|
|
|
|
{
|
|
|
|
$$.regnum = $1; $$.type = BWRITERSPR_LABEL;
|
|
|
|
}
|
2010-03-28 21:16:09 +02:00
|
|
|
|
|
|
|
%%
|
|
|
|
|
|
|
|
void asmshader_error (char const *s) {
|
|
|
|
asmparser_message(&asm_ctx, "Line %u: Error \"%s\" from bison\n", asm_ctx.line_no, s);
|
|
|
|
set_parse_status(&asm_ctx, PARSE_ERR);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Error reporting function */
|
|
|
|
void asmparser_message(struct asm_parser *ctx, const char *fmt, ...) {
|
|
|
|
va_list args;
|
|
|
|
char* newbuffer;
|
|
|
|
int rc, newsize;
|
|
|
|
|
|
|
|
if(ctx->messagecapacity == 0) {
|
|
|
|
ctx->messages = asm_alloc(MESSAGEBUFFER_INITIAL_SIZE);
|
|
|
|
if(ctx->messages == NULL) {
|
|
|
|
ERR("Error allocating memory for parser messages\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ctx->messagecapacity = MESSAGEBUFFER_INITIAL_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
while(1) {
|
|
|
|
va_start(args, fmt);
|
|
|
|
rc = vsnprintf(ctx->messages + ctx->messagesize,
|
|
|
|
ctx->messagecapacity - ctx->messagesize, fmt, args);
|
|
|
|
va_end(args);
|
|
|
|
|
|
|
|
if (rc < 0 || /* C89 */
|
|
|
|
rc >= ctx->messagecapacity - ctx->messagesize) { /* C99 */
|
|
|
|
/* Resize the buffer */
|
|
|
|
newsize = ctx->messagecapacity * 2;
|
|
|
|
newbuffer = asm_realloc(ctx->messages, newsize);
|
|
|
|
if(newbuffer == NULL){
|
|
|
|
ERR("Error reallocating memory for parser messages\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ctx->messages = newbuffer;
|
|
|
|
ctx->messagecapacity = newsize;
|
|
|
|
} else {
|
|
|
|
ctx->messagesize += rc;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* New status is the worst between current status and parameter value */
|
|
|
|
void set_parse_status(struct asm_parser *ctx, enum parse_status status) {
|
|
|
|
if(status == PARSE_ERR) ctx->status = PARSE_ERR;
|
|
|
|
else if(status == PARSE_WARN && ctx->status == PARSE_SUCCESS) ctx->status = PARSE_WARN;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct bwriter_shader *parse_asm_shader(char **messages) {
|
|
|
|
struct bwriter_shader *ret = NULL;
|
|
|
|
|
|
|
|
asm_ctx.shader = NULL;
|
|
|
|
asm_ctx.status = PARSE_SUCCESS;
|
|
|
|
asm_ctx.messagesize = asm_ctx.messagecapacity = 0;
|
|
|
|
asm_ctx.line_no = 1;
|
|
|
|
|
|
|
|
asmshader_parse();
|
|
|
|
|
|
|
|
if(asm_ctx.status != PARSE_ERR) ret = asm_ctx.shader;
|
|
|
|
else if(asm_ctx.shader) SlDeleteShader(asm_ctx.shader);
|
|
|
|
|
|
|
|
if(messages) {
|
|
|
|
if(asm_ctx.messagesize) {
|
|
|
|
/* Shrink the buffer to the used size */
|
|
|
|
*messages = asm_realloc(asm_ctx.messages, asm_ctx.messagesize + 1);
|
|
|
|
if(!*messages) {
|
|
|
|
ERR("Out of memory, no messages reported\n");
|
|
|
|
asm_free(asm_ctx.messages);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
*messages = NULL;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if(asm_ctx.messagecapacity) asm_free(asm_ctx.messages);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|