Restrições de uso da interface
O hardware de GPU atual não dá suporte a informações de slot variáveis no runtime do sombreador. Como consequência, as referências de interface não podem ser modificadas em uma expressão condicional, como uma instrução if ou switch.
O código de sombreador a seguir ilustra quando essa restrição ocorrerá e uma possível abordagem alternativa.
Dadas as seguintes declarações de interface:
interface A
{
float GetRatio();
bool IsGood();
};
interface B
{
float GetValue();
};
A arrayA[6];
B arrayB[6];
Dadas as seguintes declarações de classe:
class C1 : A
{
float var;
float GetRatio() { return 1.0f; }
bool IsGood() { return true; }
};
class C2 : C1, B
{
float GetRatio() { return C1::GetRatio() * 0.33f; }
float GetValue() { return 5.0f; }
bool IsGood() { return false; }
};
class C3 : B
{
float var;
float GetValue() { return -1.0f; }
};
class C4 : A, B
{
float var;
float GetRatio() { return var; }
float GetValue() { return var * 2.0f; }
bool IsGood() { return var > 0.0f; }
};
Uma referência de interface não pode ser modificada dentro da expressão condicional (uma instrução if):
float main() : wicked
{
float rev;
{
A a = arrayA[0];
for( uint i = 0; i < 6; ++i )
{
if( arrayA[i].IsGood() )
{
// This forces the loop to be unrolled,
// since the slot information is changing.
a = arrayA[i];
rev -= arrayA[i-2].GetRatio();
}
else
{
// This causes an error since the compiler is
// unable to determine the interface slot
rev += arrayB[i].GetValue() + a.GetRatio();
}
}
}
return rev;
}
Considerando as mesmas declarações de interface e classe, você pode usar um índice para fornecer a mesma funcionalidade e evitar o cancelamento do registro do loop forçado.
float main() : wicked
{
float rev;
{
uint index = 0;
for( uint i = 0; i < 6; ++i )
{
if( arrayA[i].IsGood() )
{
index = i;
rev -= arrayA[i-2].GetRatio();
}
else
{
rev += arrayB[i].GetValue() + arrayA[index].GetRatio();
}
}
}
return rev;
}
Tópicos relacionados