mirror of
https://gitlab.freedesktop.org/mesa/mesa.git
synced 2025-12-24 02:20:11 +01:00
What a beast. r300g doesn't depend on files from r300c anymore, so r300c is now left to its own fate. BTW 'make test' can be invoked from the gallium/r300 directory to run some compiler unit tests.
701 lines
17 KiB
C
701 lines
17 KiB
C
/*
|
|
* Copyright 2010 Tom Stellard <tstellar@gmail.com>
|
|
*
|
|
* All Rights Reserved.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files (the
|
|
* "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice (including the
|
|
* next paragraph) shall be included in all copies or substantial
|
|
* portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
* IN NO EVENT SHALL THE COPYRIGHT OWNER(S) AND/OR ITS SUPPLIERS BE
|
|
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
/**
|
|
* \file
|
|
*/
|
|
|
|
#include "radeon_compiler_util.h"
|
|
|
|
#include "radeon_compiler.h"
|
|
#include "radeon_dataflow.h"
|
|
/**
|
|
*/
|
|
unsigned int rc_swizzle_to_writemask(unsigned int swz)
|
|
{
|
|
unsigned int mask = 0;
|
|
unsigned int i;
|
|
|
|
for(i = 0; i < 4; i++) {
|
|
mask |= 1 << GET_SWZ(swz, i);
|
|
}
|
|
mask &= RC_MASK_XYZW;
|
|
|
|
return mask;
|
|
}
|
|
|
|
rc_swizzle get_swz(unsigned int swz, rc_swizzle idx)
|
|
{
|
|
if (idx & 0x4)
|
|
return idx;
|
|
return GET_SWZ(swz, idx);
|
|
}
|
|
|
|
/**
|
|
* The purpose of this function is to standardize the number channels used by
|
|
* swizzles. All swizzles regardless of what instruction they are a part of
|
|
* should have 4 channels initialized with values.
|
|
* @param channels The number of channels in initial_value that have a
|
|
* meaningful value.
|
|
* @return An initialized swizzle that has all of the unused channels set to
|
|
* RC_SWIZZLE_UNUSED.
|
|
*/
|
|
unsigned int rc_init_swizzle(unsigned int initial_value, unsigned int channels)
|
|
{
|
|
unsigned int i;
|
|
for (i = channels; i < 4; i++) {
|
|
SET_SWZ(initial_value, i, RC_SWIZZLE_UNUSED);
|
|
}
|
|
return initial_value;
|
|
}
|
|
|
|
unsigned int combine_swizzles4(unsigned int src,
|
|
rc_swizzle swz_x, rc_swizzle swz_y, rc_swizzle swz_z, rc_swizzle swz_w)
|
|
{
|
|
unsigned int ret = 0;
|
|
|
|
ret |= get_swz(src, swz_x);
|
|
ret |= get_swz(src, swz_y) << 3;
|
|
ret |= get_swz(src, swz_z) << 6;
|
|
ret |= get_swz(src, swz_w) << 9;
|
|
|
|
return ret;
|
|
}
|
|
|
|
unsigned int combine_swizzles(unsigned int src, unsigned int swz)
|
|
{
|
|
unsigned int ret = 0;
|
|
|
|
ret |= get_swz(src, GET_SWZ(swz, RC_SWIZZLE_X));
|
|
ret |= get_swz(src, GET_SWZ(swz, RC_SWIZZLE_Y)) << 3;
|
|
ret |= get_swz(src, GET_SWZ(swz, RC_SWIZZLE_Z)) << 6;
|
|
ret |= get_swz(src, GET_SWZ(swz, RC_SWIZZLE_W)) << 9;
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @param mask Must be either RC_MASK_X, RC_MASK_Y, RC_MASK_Z, or RC_MASK_W
|
|
*/
|
|
rc_swizzle rc_mask_to_swizzle(unsigned int mask)
|
|
{
|
|
switch (mask) {
|
|
case RC_MASK_X: return RC_SWIZZLE_X;
|
|
case RC_MASK_Y: return RC_SWIZZLE_Y;
|
|
case RC_MASK_Z: return RC_SWIZZLE_Z;
|
|
case RC_MASK_W: return RC_SWIZZLE_W;
|
|
}
|
|
return RC_SWIZZLE_UNUSED;
|
|
}
|
|
|
|
/* Reorder mask bits according to swizzle. */
|
|
unsigned swizzle_mask(unsigned swizzle, unsigned mask)
|
|
{
|
|
unsigned ret = 0;
|
|
for (unsigned chan = 0; chan < 4; ++chan) {
|
|
unsigned swz = GET_SWZ(swizzle, chan);
|
|
if (swz < 4)
|
|
ret |= GET_BIT(mask, swz) << chan;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static unsigned int srcs_need_rewrite(const struct rc_opcode_info * info)
|
|
{
|
|
if (info->HasTexture) {
|
|
return 0;
|
|
}
|
|
switch (info->Opcode) {
|
|
case RC_OPCODE_DP2:
|
|
case RC_OPCODE_DP3:
|
|
case RC_OPCODE_DP4:
|
|
case RC_OPCODE_DDX:
|
|
case RC_OPCODE_DDY:
|
|
return 0;
|
|
default:
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return A swizzle the results from converting old_swizzle using
|
|
* conversion_swizzle
|
|
*/
|
|
unsigned int rc_adjust_channels(
|
|
unsigned int old_swizzle,
|
|
unsigned int conversion_swizzle)
|
|
{
|
|
unsigned int i;
|
|
unsigned int new_swizzle = rc_init_swizzle(RC_SWIZZLE_UNUSED, 0);
|
|
for (i = 0; i < 4; i++) {
|
|
unsigned int new_chan = get_swz(conversion_swizzle, i);
|
|
if (new_chan == RC_SWIZZLE_UNUSED) {
|
|
continue;
|
|
}
|
|
SET_SWZ(new_swizzle, new_chan, GET_SWZ(old_swizzle, i));
|
|
}
|
|
return new_swizzle;
|
|
}
|
|
|
|
static unsigned int rewrite_writemask(
|
|
unsigned int old_mask,
|
|
unsigned int conversion_swizzle)
|
|
{
|
|
unsigned int new_mask = 0;
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < 4; i++) {
|
|
if (!GET_BIT(old_mask, i)
|
|
|| GET_SWZ(conversion_swizzle, i) == RC_SWIZZLE_UNUSED) {
|
|
continue;
|
|
}
|
|
new_mask |= (1 << GET_SWZ(conversion_swizzle, i));
|
|
}
|
|
|
|
return new_mask;
|
|
}
|
|
|
|
/**
|
|
* This function rewrites the writemask of sub and adjusts the swizzles
|
|
* of all its source registers based on the conversion_swizzle.
|
|
* conversion_swizzle represents a mapping of the old writemask to the
|
|
* new writemask. For a detailed description of how conversion swizzles
|
|
* work see rc_rewrite_swizzle().
|
|
*/
|
|
void rc_pair_rewrite_writemask(
|
|
struct rc_pair_sub_instruction * sub,
|
|
unsigned int conversion_swizzle)
|
|
{
|
|
const struct rc_opcode_info * info = rc_get_opcode_info(sub->Opcode);
|
|
unsigned int i;
|
|
|
|
sub->WriteMask = rewrite_writemask(sub->WriteMask, conversion_swizzle);
|
|
|
|
if (!srcs_need_rewrite(info)) {
|
|
return ;
|
|
}
|
|
|
|
for (i = 0; i < info->NumSrcRegs; i++) {
|
|
sub->Arg[i].Swizzle =
|
|
rc_adjust_channels(sub->Arg[i].Swizzle,
|
|
conversion_swizzle);
|
|
}
|
|
}
|
|
|
|
static void normal_rewrite_writemask_cb(
|
|
void * userdata,
|
|
struct rc_instruction * inst,
|
|
struct rc_src_register * src)
|
|
{
|
|
unsigned int * new_mask = (unsigned int *)userdata;
|
|
src->Swizzle = rc_adjust_channels(src->Swizzle, *new_mask);
|
|
}
|
|
|
|
/**
|
|
* This function is the same as rc_pair_rewrite_writemask() except it
|
|
* operates on normal instructions.
|
|
*/
|
|
void rc_normal_rewrite_writemask(
|
|
struct rc_instruction * inst,
|
|
unsigned int conversion_swizzle)
|
|
{
|
|
unsigned int new_mask;
|
|
struct rc_sub_instruction * sub = &inst->U.I;
|
|
const struct rc_opcode_info * info = rc_get_opcode_info(sub->Opcode);
|
|
sub->DstReg.WriteMask =
|
|
rewrite_writemask(sub->DstReg.WriteMask, conversion_swizzle);
|
|
|
|
if (info->HasTexture) {
|
|
unsigned int i;
|
|
assert(sub->TexSwizzle == RC_SWIZZLE_XYZW);
|
|
for (i = 0; i < 4; i++) {
|
|
unsigned int swz = GET_SWZ(conversion_swizzle, i);
|
|
if (swz > 3)
|
|
continue;
|
|
SET_SWZ(sub->TexSwizzle, swz, i);
|
|
}
|
|
}
|
|
|
|
if (!srcs_need_rewrite(info)) {
|
|
return;
|
|
}
|
|
|
|
new_mask = sub->DstReg.WriteMask;
|
|
rc_for_all_reads_src(inst, normal_rewrite_writemask_cb, &new_mask);
|
|
}
|
|
|
|
/**
|
|
* This function replaces each value 'swz' in swizzle with the value of
|
|
* GET_SWZ(conversion_swizzle, swz). So, if you want to change all the X's
|
|
* in swizzle to Y, then conversion_swizzle should be Y___ (0xff9). If you want
|
|
* to change all the Y's in swizzle to X, then conversion_swizzle should be
|
|
* _X__ (0xfc7). If you want to change the Y's to X and the X's to Y, then
|
|
* conversion swizzle should be YX__ (0xfc1).
|
|
* @param swizzle The swizzle to change
|
|
* @param conversion_swizzle Describes the conversion to perform on the swizzle
|
|
* @return A converted swizzle
|
|
*/
|
|
unsigned int rc_rewrite_swizzle(
|
|
unsigned int swizzle,
|
|
unsigned int conversion_swizzle)
|
|
{
|
|
unsigned int chan;
|
|
unsigned int out_swizzle = swizzle;
|
|
|
|
for (chan = 0; chan < 4; chan++) {
|
|
unsigned int swz = GET_SWZ(swizzle, chan);
|
|
unsigned int new_swz;
|
|
if (swz > 3) {
|
|
SET_SWZ(out_swizzle, chan, swz);
|
|
} else {
|
|
new_swz = GET_SWZ(conversion_swizzle, swz);
|
|
if (new_swz != RC_SWIZZLE_UNUSED) {
|
|
SET_SWZ(out_swizzle, chan, new_swz);
|
|
} else {
|
|
SET_SWZ(out_swizzle, chan, swz);
|
|
}
|
|
}
|
|
}
|
|
return out_swizzle;
|
|
}
|
|
|
|
/**
|
|
* Left multiplication of a register with a swizzle
|
|
*/
|
|
struct rc_src_register lmul_swizzle(unsigned int swizzle, struct rc_src_register srcreg)
|
|
{
|
|
struct rc_src_register tmp = srcreg;
|
|
int i;
|
|
tmp.Swizzle = 0;
|
|
tmp.Negate = 0;
|
|
for(i = 0; i < 4; ++i) {
|
|
rc_swizzle swz = GET_SWZ(swizzle, i);
|
|
if (swz < 4) {
|
|
tmp.Swizzle |= GET_SWZ(srcreg.Swizzle, swz) << (i*3);
|
|
tmp.Negate |= GET_BIT(srcreg.Negate, swz) << i;
|
|
} else {
|
|
tmp.Swizzle |= swz << (i*3);
|
|
}
|
|
}
|
|
return tmp;
|
|
}
|
|
|
|
void reset_srcreg(struct rc_src_register* reg)
|
|
{
|
|
memset(reg, 0, sizeof(struct rc_src_register));
|
|
reg->Swizzle = RC_SWIZZLE_XYZW;
|
|
}
|
|
|
|
unsigned int rc_src_reads_dst_mask(
|
|
rc_register_file src_file,
|
|
unsigned int src_idx,
|
|
unsigned int src_swz,
|
|
rc_register_file dst_file,
|
|
unsigned int dst_idx,
|
|
unsigned int dst_mask)
|
|
{
|
|
if (src_file != dst_file || src_idx != dst_idx) {
|
|
return RC_MASK_NONE;
|
|
}
|
|
return dst_mask & rc_swizzle_to_writemask(src_swz);
|
|
}
|
|
|
|
/**
|
|
* @return A bit mask specifying whether this swizzle will select from an RGB
|
|
* source, an Alpha source, or both.
|
|
*/
|
|
unsigned int rc_source_type_swz(unsigned int swizzle)
|
|
{
|
|
unsigned int chan;
|
|
unsigned int swz = RC_SWIZZLE_UNUSED;
|
|
unsigned int ret = RC_SOURCE_NONE;
|
|
|
|
for(chan = 0; chan < 4; chan++) {
|
|
swz = GET_SWZ(swizzle, chan);
|
|
if (swz == RC_SWIZZLE_W) {
|
|
ret |= RC_SOURCE_ALPHA;
|
|
} else if (swz == RC_SWIZZLE_X || swz == RC_SWIZZLE_Y
|
|
|| swz == RC_SWIZZLE_Z) {
|
|
ret |= RC_SOURCE_RGB;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
unsigned int rc_source_type_mask(unsigned int mask)
|
|
{
|
|
unsigned int ret = RC_SOURCE_NONE;
|
|
|
|
if (mask & RC_MASK_XYZ)
|
|
ret |= RC_SOURCE_RGB;
|
|
|
|
if (mask & RC_MASK_W)
|
|
ret |= RC_SOURCE_ALPHA;
|
|
|
|
return ret;
|
|
}
|
|
|
|
struct src_select {
|
|
rc_register_file File;
|
|
int Index;
|
|
unsigned int SrcType;
|
|
};
|
|
|
|
struct can_use_presub_data {
|
|
struct src_select Selects[5];
|
|
unsigned int SelectCount;
|
|
const struct rc_src_register * ReplaceReg;
|
|
unsigned int ReplaceRemoved;
|
|
};
|
|
|
|
static void can_use_presub_data_add_select(
|
|
struct can_use_presub_data * data,
|
|
rc_register_file file,
|
|
unsigned int index,
|
|
unsigned int src_type)
|
|
{
|
|
struct src_select * select;
|
|
|
|
select = &data->Selects[data->SelectCount++];
|
|
select->File = file;
|
|
select->Index = index;
|
|
select->SrcType = src_type;
|
|
}
|
|
|
|
/**
|
|
* This callback function counts the number of sources in inst that are
|
|
* different from the sources in can_use_presub_data->RemoveSrcs.
|
|
*/
|
|
static void can_use_presub_read_cb(
|
|
void * userdata,
|
|
struct rc_instruction * inst,
|
|
struct rc_src_register * src)
|
|
{
|
|
struct can_use_presub_data * d = userdata;
|
|
|
|
if (!d->ReplaceRemoved && src == d->ReplaceReg) {
|
|
d->ReplaceRemoved = 1;
|
|
return;
|
|
}
|
|
|
|
if (src->File == RC_FILE_NONE)
|
|
return;
|
|
|
|
can_use_presub_data_add_select(d, src->File, src->Index,
|
|
rc_source_type_swz(src->Swizzle));
|
|
}
|
|
|
|
unsigned int rc_inst_can_use_presub(
|
|
struct rc_instruction * inst,
|
|
rc_presubtract_op presub_op,
|
|
unsigned int presub_writemask,
|
|
const struct rc_src_register * replace_reg,
|
|
const struct rc_src_register * presub_src0,
|
|
const struct rc_src_register * presub_src1)
|
|
{
|
|
struct can_use_presub_data d;
|
|
unsigned int num_presub_srcs;
|
|
unsigned int i;
|
|
const struct rc_opcode_info * info =
|
|
rc_get_opcode_info(inst->U.I.Opcode);
|
|
int rgb_count = 0, alpha_count = 0;
|
|
unsigned int src_type0, src_type1;
|
|
|
|
if (presub_op == RC_PRESUB_NONE) {
|
|
return 1;
|
|
}
|
|
|
|
if (info->HasTexture) {
|
|
return 0;
|
|
}
|
|
|
|
/* We can't use more than one presubtract value in an
|
|
* instruction, unless the two prsubtract operations
|
|
* are the same and read from the same registers.
|
|
* XXX For now we will limit instructions to only one presubtract
|
|
* value.*/
|
|
if (inst->U.I.PreSub.Opcode != RC_PRESUB_NONE) {
|
|
return 0;
|
|
}
|
|
|
|
memset(&d, 0, sizeof(d));
|
|
d.ReplaceReg = replace_reg;
|
|
|
|
rc_for_all_reads_src(inst, can_use_presub_read_cb, &d);
|
|
|
|
num_presub_srcs = rc_presubtract_src_reg_count(presub_op);
|
|
|
|
src_type0 = rc_source_type_swz(presub_src0->Swizzle);
|
|
can_use_presub_data_add_select(&d,
|
|
presub_src0->File,
|
|
presub_src0->Index,
|
|
src_type0);
|
|
|
|
if (num_presub_srcs > 1) {
|
|
src_type1 = rc_source_type_swz(presub_src1->Swizzle);
|
|
can_use_presub_data_add_select(&d,
|
|
presub_src1->File,
|
|
presub_src1->Index,
|
|
src_type1);
|
|
|
|
/* Even if both of the presub sources read from the same
|
|
* register, we still need to use 2 different source selects
|
|
* for them, so we need to increment the count to compensate.
|
|
*/
|
|
if (presub_src0->File == presub_src1->File
|
|
&& presub_src0->Index == presub_src1->Index) {
|
|
if (src_type0 & src_type1 & RC_SOURCE_RGB) {
|
|
rgb_count++;
|
|
}
|
|
if (src_type0 & src_type1 & RC_SOURCE_ALPHA) {
|
|
alpha_count++;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Count the number of source selects for Alpha and RGB. If we
|
|
* encounter two of the same source selects then we can ignore the
|
|
* first one. */
|
|
for (i = 0; i < d.SelectCount; i++) {
|
|
unsigned int j;
|
|
unsigned int src_type = d.Selects[i].SrcType;
|
|
for (j = i + 1; j < d.SelectCount; j++) {
|
|
if (d.Selects[i].File == d.Selects[j].File
|
|
&& d.Selects[i].Index == d.Selects[j].Index) {
|
|
src_type &= ~d.Selects[j].SrcType;
|
|
}
|
|
}
|
|
if (src_type & RC_SOURCE_RGB) {
|
|
rgb_count++;
|
|
}
|
|
|
|
if (src_type & RC_SOURCE_ALPHA) {
|
|
alpha_count++;
|
|
}
|
|
}
|
|
|
|
if (rgb_count > 3 || alpha_count > 3) {
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
struct max_data {
|
|
unsigned int Max;
|
|
unsigned int HasFileType;
|
|
rc_register_file File;
|
|
};
|
|
|
|
static void max_callback(
|
|
void * userdata,
|
|
struct rc_instruction * inst,
|
|
rc_register_file file,
|
|
unsigned int index,
|
|
unsigned int mask)
|
|
{
|
|
struct max_data * d = (struct max_data*)userdata;
|
|
if (file == d->File && (!d->HasFileType || index > d->Max)) {
|
|
d->Max = index;
|
|
d->HasFileType = 1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @return The maximum index of the specified register file used by the
|
|
* program.
|
|
*/
|
|
int rc_get_max_index(
|
|
struct radeon_compiler * c,
|
|
rc_register_file file)
|
|
{
|
|
struct max_data data;
|
|
struct rc_instruction * inst;
|
|
data.Max = 0;
|
|
data.HasFileType = 0;
|
|
data.File = file;
|
|
for (inst = c->Program.Instructions.Next;
|
|
inst != &c->Program.Instructions;
|
|
inst = inst->Next) {
|
|
rc_for_all_reads_mask(inst, max_callback, &data);
|
|
rc_for_all_writes_mask(inst, max_callback, &data);
|
|
}
|
|
if (!data.HasFileType) {
|
|
return -1;
|
|
} else {
|
|
return data.Max;
|
|
}
|
|
}
|
|
|
|
static unsigned int get_source_readmask(
|
|
struct rc_pair_sub_instruction * sub,
|
|
unsigned int source,
|
|
unsigned int src_type)
|
|
{
|
|
unsigned int i;
|
|
unsigned int readmask = 0;
|
|
const struct rc_opcode_info * info = rc_get_opcode_info(sub->Opcode);
|
|
|
|
for (i = 0; i < info->NumSrcRegs; i++) {
|
|
if (sub->Arg[i].Source != source
|
|
|| src_type != rc_source_type_swz(sub->Arg[i].Swizzle)) {
|
|
continue;
|
|
}
|
|
readmask |= rc_swizzle_to_writemask(sub->Arg[i].Swizzle);
|
|
}
|
|
return readmask;
|
|
}
|
|
|
|
/**
|
|
* This function attempts to remove a source from a pair instructions.
|
|
* @param inst
|
|
* @param src_type RC_SOURCE_RGB, RC_SOURCE_ALPHA, or both bitwise or'd
|
|
* @param source The index of the source to remove
|
|
* @param new_readmask A mask representing the components that are read by
|
|
* the source that is intended to replace the one you are removing. If you
|
|
* want to remove a source only and not replace it, this parameter should be
|
|
* zero.
|
|
* @return 1 if the source was successfully removed, 0 if it was not
|
|
*/
|
|
unsigned int rc_pair_remove_src(
|
|
struct rc_instruction * inst,
|
|
unsigned int src_type,
|
|
unsigned int source,
|
|
unsigned int new_readmask)
|
|
{
|
|
unsigned int readmask = 0;
|
|
|
|
readmask |= get_source_readmask(&inst->U.P.RGB, source, src_type);
|
|
readmask |= get_source_readmask(&inst->U.P.Alpha, source, src_type);
|
|
|
|
if ((new_readmask & readmask) != readmask)
|
|
return 0;
|
|
|
|
if (src_type & RC_SOURCE_RGB) {
|
|
memset(&inst->U.P.RGB.Src[source], 0,
|
|
sizeof(struct rc_pair_instruction_source));
|
|
}
|
|
|
|
if (src_type & RC_SOURCE_ALPHA) {
|
|
memset(&inst->U.P.Alpha.Src[source], 0,
|
|
sizeof(struct rc_pair_instruction_source));
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
* @return RC_OPCODE_NOOP if inst is not a flow control instruction.
|
|
* @return The opcode of inst if it is a flow control instruction.
|
|
*/
|
|
rc_opcode rc_get_flow_control_inst(struct rc_instruction * inst)
|
|
{
|
|
const struct rc_opcode_info * info;
|
|
if (inst->Type == RC_INSTRUCTION_NORMAL) {
|
|
info = rc_get_opcode_info(inst->U.I.Opcode);
|
|
} else {
|
|
info = rc_get_opcode_info(inst->U.P.RGB.Opcode);
|
|
/*A flow control instruction shouldn't have an alpha
|
|
* instruction.*/
|
|
assert(!info->IsFlowControl ||
|
|
inst->U.P.Alpha.Opcode == RC_OPCODE_NOP);
|
|
}
|
|
|
|
if (info->IsFlowControl)
|
|
return info->Opcode;
|
|
else
|
|
return RC_OPCODE_NOP;
|
|
|
|
}
|
|
|
|
/**
|
|
* @return The BGNLOOP instruction that starts the loop ended by endloop.
|
|
*/
|
|
struct rc_instruction * rc_match_endloop(struct rc_instruction * endloop)
|
|
{
|
|
unsigned int endloop_count = 0;
|
|
struct rc_instruction * inst;
|
|
for (inst = endloop->Prev; inst != endloop; inst = inst->Prev) {
|
|
rc_opcode op = rc_get_flow_control_inst(inst);
|
|
if (op == RC_OPCODE_ENDLOOP) {
|
|
endloop_count++;
|
|
} else if (op == RC_OPCODE_BGNLOOP) {
|
|
if (endloop_count == 0) {
|
|
return inst;
|
|
} else {
|
|
endloop_count--;
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* @return The ENDLOOP instruction that ends the loop started by bgnloop.
|
|
*/
|
|
struct rc_instruction * rc_match_bgnloop(struct rc_instruction * bgnloop)
|
|
{
|
|
unsigned int bgnloop_count = 0;
|
|
struct rc_instruction * inst;
|
|
for (inst = bgnloop->Next; inst!=bgnloop; inst = inst->Next) {
|
|
rc_opcode op = rc_get_flow_control_inst(inst);
|
|
if (op == RC_OPCODE_BGNLOOP) {
|
|
bgnloop_count++;
|
|
} else if (op == RC_OPCODE_ENDLOOP) {
|
|
if (bgnloop_count == 0) {
|
|
return inst;
|
|
} else {
|
|
bgnloop_count--;
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* @return A conversion swizzle for converting from old_mask->new_mask
|
|
*/
|
|
unsigned int rc_make_conversion_swizzle(
|
|
unsigned int old_mask,
|
|
unsigned int new_mask)
|
|
{
|
|
unsigned int conversion_swizzle = rc_init_swizzle(RC_SWIZZLE_UNUSED, 0);
|
|
unsigned int old_idx;
|
|
unsigned int new_idx = 0;
|
|
for (old_idx = 0; old_idx < 4; old_idx++) {
|
|
if (!GET_BIT(old_mask, old_idx))
|
|
continue;
|
|
for ( ; new_idx < 4; new_idx++) {
|
|
if (GET_BIT(new_mask, new_idx)) {
|
|
SET_SWZ(conversion_swizzle, old_idx, new_idx);
|
|
new_idx++;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
return conversion_swizzle;
|
|
}
|