1 /**************************************************************************
3 * Copyright 2008 Tungsten Graphics, Inc., Cedar Park, Texas.
6 * Permission is hereby granted, free of charge, to any person obtaining a
7 * copy of this software and associated documentation files (the
8 * "Software"), to deal in the Software without restriction, including
9 * without limitation the rights to use, copy, modify, merge, publish,
10 * distribute, sub license, and/or sell copies of the Software, and to
11 * permit persons to whom the Software is furnished to do so, subject to
12 * the following conditions:
14 * The above copyright notice and this permission notice (including the
15 * next paragraph) shall be included in all copies or substantial portions
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT.
21 * IN NO EVENT SHALL TUNGSTEN GRAPHICS AND/OR ITS SUPPLIERS BE LIABLE FOR
22 * ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
23 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
24 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
26 **************************************************************************/
29 * TGSI program transformation utility.
34 #include "util/u_debug.h"
36 #include "tgsi_transform.h"
41 emit_instruction(struct tgsi_transform_context
*ctx
,
42 const struct tgsi_full_instruction
*inst
)
46 ti
+= tgsi_build_full_instruction(inst
,
49 ctx
->max_tokens_out
- ti
);
55 emit_declaration(struct tgsi_transform_context
*ctx
,
56 const struct tgsi_full_declaration
*decl
)
60 ti
+= tgsi_build_full_declaration(decl
,
63 ctx
->max_tokens_out
- ti
);
69 emit_immediate(struct tgsi_transform_context
*ctx
,
70 const struct tgsi_full_immediate
*imm
)
74 ti
+= tgsi_build_full_immediate(imm
,
77 ctx
->max_tokens_out
- ti
);
83 emit_property(struct tgsi_transform_context
*ctx
,
84 const struct tgsi_full_property
*prop
)
88 ti
+= tgsi_build_full_property(prop
,
91 ctx
->max_tokens_out
- ti
);
97 * Apply user-defined transformations to the input shader to produce
99 * For example, a register search-and-replace operation could be applied
100 * by defining a transform_instruction() callback that examined and changed
101 * the instruction src/dest regs.
103 * \return number of tokens emitted
106 tgsi_transform_shader(const struct tgsi_token
*tokens_in
,
107 struct tgsi_token
*tokens_out
,
109 struct tgsi_transform_context
*ctx
)
114 struct tgsi_parse_context parse
;
117 struct tgsi_processor
*processor
;
121 ** callback context init
123 ctx
->emit_instruction
= emit_instruction
;
124 ctx
->emit_declaration
= emit_declaration
;
125 ctx
->emit_immediate
= emit_immediate
;
126 ctx
->emit_property
= emit_property
;
127 ctx
->tokens_out
= tokens_out
;
128 ctx
->max_tokens_out
= max_tokens_out
;
132 ** Setup to begin parsing input shader
134 if (tgsi_parse_init( &parse
, tokens_in
) != TGSI_PARSE_OK
) {
135 debug_printf("tgsi_parse_init() failed in tgsi_transform_shader()!\n");
138 procType
= parse
.FullHeader
.Processor
.Processor
;
139 assert(procType
== TGSI_PROCESSOR_FRAGMENT
||
140 procType
== TGSI_PROCESSOR_VERTEX
||
141 procType
== TGSI_PROCESSOR_GEOMETRY
);
145 ** Setup output shader
147 ctx
->header
= (struct tgsi_header
*)tokens_out
;
148 *ctx
->header
= tgsi_build_header();
150 processor
= (struct tgsi_processor
*) (tokens_out
+ 1);
151 *processor
= tgsi_build_processor( procType
, ctx
->header
);
157 ** Loop over incoming program tokens/instructions
159 while( !tgsi_parse_end_of_tokens( &parse
) ) {
161 tgsi_parse_token( &parse
);
163 switch( parse
.FullToken
.Token
.Type
) {
164 case TGSI_TOKEN_TYPE_INSTRUCTION
:
166 struct tgsi_full_instruction
*fullinst
167 = &parse
.FullToken
.FullInstruction
;
169 if (ctx
->transform_instruction
)
170 ctx
->transform_instruction(ctx
, fullinst
);
172 ctx
->emit_instruction(ctx
, fullinst
);
176 case TGSI_TOKEN_TYPE_DECLARATION
:
178 struct tgsi_full_declaration
*fulldecl
179 = &parse
.FullToken
.FullDeclaration
;
181 if (ctx
->transform_declaration
)
182 ctx
->transform_declaration(ctx
, fulldecl
);
184 ctx
->emit_declaration(ctx
, fulldecl
);
188 case TGSI_TOKEN_TYPE_IMMEDIATE
:
190 struct tgsi_full_immediate
*fullimm
191 = &parse
.FullToken
.FullImmediate
;
193 if (ctx
->transform_immediate
)
194 ctx
->transform_immediate(ctx
, fullimm
);
196 ctx
->emit_immediate(ctx
, fullimm
);
199 case TGSI_TOKEN_TYPE_PROPERTY
:
201 struct tgsi_full_property
*fullprop
202 = &parse
.FullToken
.FullProperty
;
204 if (ctx
->transform_property
)
205 ctx
->transform_property(ctx
, fullprop
);
207 ctx
->emit_property(ctx
, fullprop
);
220 tgsi_parse_free (&parse
);
226 #include "tgsi_text.h"
228 extern int tgsi_transform_foo( struct tgsi_token
*tokens_out
,
229 uint max_tokens_out
);
231 /* This function exists only so that tgsi_text_translate() doesn't get
232 * magic-ed out of the libtgsi.a archive by the build system. Don't
233 * remove unless you know this has been fixed - check on mingw/scons
237 tgsi_transform_foo( struct tgsi_token
*tokens_out
,
238 uint max_tokens_out
)
242 "DCL IN[0], COLOR, CONSTANT\n"
243 "DCL OUT[0], COLOR\n"
244 " 0: MOV OUT[0], IN[0]\n"
247 return tgsi_text_translate( text
,