r600: add support for CUBE textures, also TXP

seems to work here ...
This commit is contained in:
Andre Maasikas
2009-09-23 14:20:59 +03:00
committed by Alex Deucher
parent 639fb1472d
commit 2058dfaa47
2 changed files with 263 additions and 47 deletions

View File

@@ -213,7 +213,7 @@ GLboolean is_reduction_opcode(PVSDWORD* dest)
{ {
if (dest->dst.op3 == 0) if (dest->dst.op3 == 0)
{ {
if ( (dest->dst.opcode == SQ_OP2_INST_DOT4 || dest->dst.opcode == SQ_OP2_INST_DOT4_IEEE) ) if ( (dest->dst.opcode == SQ_OP2_INST_DOT4 || dest->dst.opcode == SQ_OP2_INST_DOT4_IEEE || dest->dst.opcode == SQ_OP2_INST_CUBE) )
{ {
return GL_TRUE; return GL_TRUE;
} }
@@ -350,6 +350,7 @@ unsigned int r700GetNumOperands(r700_AssemblerBase* pAsm)
case SQ_OP2_INST_PRED_SETNE: case SQ_OP2_INST_PRED_SETNE:
case SQ_OP2_INST_DOT4: case SQ_OP2_INST_DOT4:
case SQ_OP2_INST_DOT4_IEEE: case SQ_OP2_INST_DOT4_IEEE:
case SQ_OP2_INST_CUBE:
return 2; return 2;
case SQ_OP2_INST_MOV: case SQ_OP2_INST_MOV:
@@ -469,6 +470,9 @@ int Init_r700_AssemblerBase(SHADER_PIPE_TYPE spt, r700_AssemblerBase* pAsm, R700
pAsm->number_of_inputs = 0; pAsm->number_of_inputs = 0;
pAsm->is_tex = GL_FALSE;
pAsm->need_tex_barrier = GL_FALSE;
return 0; return 0;
} }
@@ -682,7 +686,7 @@ GLboolean add_tex_instruction(r700_AssemblerBase* pAsm,
// If this clause constains any TEX instruction that is dependent on a previous instruction, // If this clause constains any TEX instruction that is dependent on a previous instruction,
// set the barrier bit // set the barrier bit
if( pAsm->pInstDeps[pAsm->uiCurInst].nDstDep > (-1) ) if( pAsm->pInstDeps[pAsm->uiCurInst].nDstDep > (-1) || pAsm->need_tex_barrier == GL_TRUE )
{ {
pAsm->cf_current_tex_clause_ptr->m_Word1.f.barrier = 0x1; pAsm->cf_current_tex_clause_ptr->m_Word1.f.barrier = 0x1;
} }
@@ -1152,42 +1156,48 @@ GLboolean tex_src(r700_AssemblerBase *pAsm)
GLboolean bValidTexCoord = GL_FALSE; GLboolean bValidTexCoord = GL_FALSE;
if(pAsm->aArgSubst[1] >= 0)
{
bValidTexCoord = GL_TRUE;
setaddrmode_PVSSRC(&(pAsm->S[0].src), ADDR_ABSOLUTE);
pAsm->S[0].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[0].src.reg = pAsm->aArgSubst[1];
}
else
{
switch (pILInst->SrcReg[0].File) { switch (pILInst->SrcReg[0].File) {
case PROGRAM_CONSTANT: case PROGRAM_CONSTANT:
case PROGRAM_LOCAL_PARAM: case PROGRAM_LOCAL_PARAM:
case PROGRAM_ENV_PARAM: case PROGRAM_ENV_PARAM:
case PROGRAM_STATE_VAR: case PROGRAM_STATE_VAR:
bValidTexCoord = GL_TRUE; break;
setaddrmode_PVSSRC(&(pAsm->S[0].src), ADDR_ABSOLUTE); case PROGRAM_TEMPORARY:
pAsm->S[0].src.rtype = SRC_REG_TEMPORARY; bValidTexCoord = GL_TRUE;
pAsm->S[0].src.reg = pAsm->aArgSubst[1]; pAsm->S[0].src.reg = pILInst->SrcReg[0].Index +
break; pAsm->starting_temp_register_number;
case PROGRAM_TEMPORARY: pAsm->S[0].src.rtype = SRC_REG_TEMPORARY;
bValidTexCoord = GL_TRUE; break;
pAsm->S[0].src.reg = pILInst->SrcReg[0].Index + case PROGRAM_INPUT:
pAsm->starting_temp_register_number; switch (pILInst->SrcReg[0].Index)
pAsm->S[0].src.rtype = SRC_REG_TEMPORARY; {
break; case FRAG_ATTRIB_COL0:
case PROGRAM_INPUT: case FRAG_ATTRIB_COL1:
switch (pILInst->SrcReg[0].Index) case FRAG_ATTRIB_TEX0:
{ case FRAG_ATTRIB_TEX1:
case FRAG_ATTRIB_COL0: case FRAG_ATTRIB_TEX2:
case FRAG_ATTRIB_COL1: case FRAG_ATTRIB_TEX3:
case FRAG_ATTRIB_TEX0: case FRAG_ATTRIB_TEX4:
case FRAG_ATTRIB_TEX1: case FRAG_ATTRIB_TEX5:
case FRAG_ATTRIB_TEX2: case FRAG_ATTRIB_TEX6:
case FRAG_ATTRIB_TEX3: case FRAG_ATTRIB_TEX7:
case FRAG_ATTRIB_TEX4: bValidTexCoord = GL_TRUE;
case FRAG_ATTRIB_TEX5: pAsm->S[0].src.reg =
case FRAG_ATTRIB_TEX6: pAsm->uiFP_AttributeMap[pILInst->SrcReg[0].Index];
case FRAG_ATTRIB_TEX7: pAsm->S[0].src.rtype = SRC_REG_INPUT;
bValidTexCoord = GL_TRUE; break;
pAsm->S[0].src.reg = }
pAsm->uiFP_AttributeMap[pILInst->SrcReg[0].Index]; break;
pAsm->S[0].src.rtype = SRC_REG_INPUT; }
break;
}
break;
} }
if(GL_TRUE == bValidTexCoord) if(GL_TRUE == bValidTexCoord)
@@ -1956,6 +1966,8 @@ GLboolean assemble_alu_instruction(r700_AssemblerBase *pAsm)
is_single_scalar_operation = GL_FALSE; is_single_scalar_operation = GL_FALSE;
number_of_scalar_operations = 4; number_of_scalar_operations = 4;
/* current assembler doesn't do more than 1 register per source */
#if 0
/* check read port, only very preliminary algorithm, not count in /* check read port, only very preliminary algorithm, not count in
src0/1 same comp case and prev slot repeat case; also not count relative src0/1 same comp case and prev slot repeat case; also not count relative
addressing. TODO: improve performance. */ addressing. TODO: improve performance. */
@@ -1990,6 +2002,7 @@ GLboolean assemble_alu_instruction(r700_AssemblerBase *pAsm)
{ {
bSplitInst = GL_TRUE; bSplitInst = GL_TRUE;
} }
#endif
} }
contiguous_slots_needed = 0; contiguous_slots_needed = 0;
@@ -2210,9 +2223,7 @@ GLboolean next_ins(r700_AssemblerBase *pAsm)
{ {
struct prog_instruction *pILInst = &(pAsm->pILInst[pAsm->uiCurInst]); struct prog_instruction *pILInst = &(pAsm->pILInst[pAsm->uiCurInst]);
if( GL_TRUE == IsTex(pILInst->Opcode) && if( GL_TRUE == pAsm->is_tex )
/* handle const moves to temp register */
!(pAsm->D.dst.opcode == SQ_OP2_INST_MOV) )
{ {
if (pILInst->TexSrcTarget == TEXTURE_RECT_INDEX) { if (pILInst->TexSrcTarget == TEXTURE_RECT_INDEX) {
if( GL_FALSE == assemble_tex_instruction(pAsm, GL_FALSE) ) if( GL_FALSE == assemble_tex_instruction(pAsm, GL_FALSE) )
@@ -2256,7 +2267,8 @@ GLboolean next_ins(r700_AssemblerBase *pAsm)
pAsm->S[0].bits = 0; pAsm->S[0].bits = 0;
pAsm->S[1].bits = 0; pAsm->S[1].bits = 0;
pAsm->S[2].bits = 0; pAsm->S[2].bits = 0;
pAsm->is_tex = GL_FALSE;
pAsm->need_tex_barrier = GL_FALSE;
return GL_TRUE; return GL_TRUE;
} }
@@ -3379,6 +3391,9 @@ GLboolean assemble_STP(r700_AssemblerBase *pAsm)
GLboolean assemble_TEX(r700_AssemblerBase *pAsm) GLboolean assemble_TEX(r700_AssemblerBase *pAsm)
{ {
GLboolean src_const; GLboolean src_const;
GLboolean need_barrier = GL_FALSE;
checkop1(pAsm);
switch (pAsm->pILInst[pAsm->uiCurInst].SrcReg[0].File) switch (pAsm->pILInst[pAsm->uiCurInst].SrcReg[0].File)
{ {
@@ -3399,20 +3414,18 @@ GLboolean assemble_TEX(r700_AssemblerBase *pAsm)
{ {
if ( GL_FALSE == mov_temp(pAsm, 0) ) if ( GL_FALSE == mov_temp(pAsm, 0) )
return GL_FALSE; return GL_FALSE;
need_barrier = GL_TRUE;
} }
switch (pAsm->pILInst[pAsm->uiCurInst].Opcode) switch (pAsm->pILInst[pAsm->uiCurInst].Opcode)
{ {
case OPCODE_TEX: case OPCODE_TEX:
pAsm->D.dst.opcode = SQ_TEX_INST_SAMPLE;
break; break;
case OPCODE_TXB: case OPCODE_TXB:
radeon_error("do not support TXB yet\n"); radeon_error("do not support TXB yet\n");
return GL_FALSE; return GL_FALSE;
break; break;
case OPCODE_TXP: case OPCODE_TXP:
/* TODO : tex proj version : divid first 3 components by 4th */
pAsm->D.dst.opcode = SQ_TEX_INST_SAMPLE;
break; break;
default: default:
radeon_error("Internal error: bad texture op (not TEX)\n"); radeon_error("Internal error: bad texture op (not TEX)\n");
@@ -3420,6 +3433,190 @@ GLboolean assemble_TEX(r700_AssemblerBase *pAsm)
break; break;
} }
if (pAsm->pILInst[pAsm->uiCurInst].Opcode == OPCODE_TXP)
{
GLuint tmp = gethelpr(pAsm);
pAsm->D.dst.opcode = SQ_OP2_INST_RECIP_IEEE;
pAsm->D.dst.math = 1;
setaddrmode_PVSDST(&(pAsm->D.dst), ADDR_ABSOLUTE);
pAsm->D.dst.rtype = DST_REG_TEMPORARY;
pAsm->D.dst.reg = tmp;
pAsm->D.dst.writew = 1;
if( GL_FALSE == assemble_src(pAsm, 0, -1) )
{
return GL_FALSE;
}
swizzleagain_PVSSRC(&(pAsm->S[0].src), SQ_SEL_W, SQ_SEL_W, SQ_SEL_W, SQ_SEL_W);
if( GL_FALSE == next_ins(pAsm) )
{
return GL_FALSE;
}
pAsm->D.dst.opcode = SQ_OP2_INST_MUL;
setaddrmode_PVSDST(&(pAsm->D.dst), ADDR_ABSOLUTE);
pAsm->D.dst.rtype = DST_REG_TEMPORARY;
pAsm->D.dst.reg = tmp;
pAsm->D.dst.writex = 1;
pAsm->D.dst.writey = 1;
pAsm->D.dst.writez = 1;
pAsm->D.dst.writew = 0;
if( GL_FALSE == assemble_src(pAsm, 0, -1) )
{
return GL_FALSE;
}
setaddrmode_PVSSRC(&(pAsm->S[1].src), ADDR_ABSOLUTE);
pAsm->S[1].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[1].src.reg = tmp;
setswizzle_PVSSRC(&(pAsm->S[1].src), SQ_SEL_W);
if( GL_FALSE == next_ins(pAsm) )
{
return GL_FALSE;
}
pAsm->aArgSubst[1] = tmp;
need_barrier = GL_TRUE;
}
if (pAsm->pILInst[pAsm->uiCurInst].TexSrcTarget == TEXTURE_CUBE_INDEX )
{
GLuint tmp1 = gethelpr(pAsm);
GLuint tmp2 = gethelpr(pAsm);
/* tmp1.xyzw = CUBE(R0.zzxy, R0.yxzz) */
pAsm->D.dst.opcode = SQ_OP2_INST_CUBE;
setaddrmode_PVSDST(&(pAsm->D.dst), ADDR_ABSOLUTE);
pAsm->D.dst.rtype = DST_REG_TEMPORARY;
pAsm->D.dst.reg = tmp1;
nomask_PVSDST(&(pAsm->D.dst));
if( GL_FALSE == assemble_src(pAsm, 0, -1) )
{
return GL_FALSE;
}
if( GL_FALSE == assemble_src(pAsm, 0, 1) )
{
return GL_FALSE;
}
swizzleagain_PVSSRC(&(pAsm->S[0].src), SQ_SEL_Z, SQ_SEL_Z, SQ_SEL_X, SQ_SEL_Y);
swizzleagain_PVSSRC(&(pAsm->S[1].src), SQ_SEL_Y, SQ_SEL_X, SQ_SEL_Z, SQ_SEL_Z);
if( GL_FALSE == next_ins(pAsm) )
{
return GL_FALSE;
}
/* tmp1.z = ABS(tmp1.z) dont have abs support in assembler currently
* have to do explicit instruction
*/
pAsm->D.dst.opcode = SQ_OP2_INST_MAX;
setaddrmode_PVSDST(&(pAsm->D.dst), ADDR_ABSOLUTE);
pAsm->D.dst.rtype = DST_REG_TEMPORARY;
pAsm->D.dst.reg = tmp1;
pAsm->D.dst.writez = 1;
setaddrmode_PVSSRC(&(pAsm->S[0].src), ADDR_ABSOLUTE);
pAsm->S[0].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[0].src.reg = tmp1;
noswizzle_PVSSRC(&(pAsm->S[0].src));
pAsm->S[1].bits = pAsm->S[0].bits;
flipneg_PVSSRC(&(pAsm->S[1].src));
next_ins(pAsm);
/* tmp1.z = RCP_e(|tmp1.z|) */
pAsm->D.dst.opcode = SQ_OP2_INST_RECIP_IEEE;
pAsm->D.dst.math = 1;
setaddrmode_PVSDST(&(pAsm->D.dst), ADDR_ABSOLUTE);
pAsm->D.dst.rtype = DST_REG_TEMPORARY;
pAsm->D.dst.reg = tmp1;
pAsm->D.dst.writez = 1;
setaddrmode_PVSSRC(&(pAsm->S[0].src), ADDR_ABSOLUTE);
pAsm->S[0].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[0].src.reg = tmp1;
pAsm->S[0].src.swizzlex = SQ_SEL_Z;
next_ins(pAsm);
/* MULADD R0.x, R0.x, PS1, (0x3FC00000, 1.5f).x
* MULADD R0.y, R0.y, PS1, (0x3FC00000, 1.5f).x
* muladd has no writemask, have to use another temp
* also no support for imm constants, so add 1 here
*/
pAsm->D.dst.opcode = SQ_OP3_INST_MULADD;
pAsm->D.dst.op3 = 1;
setaddrmode_PVSDST(&(pAsm->D.dst), ADDR_ABSOLUTE);
pAsm->D.dst.rtype = DST_REG_TEMPORARY;
pAsm->D.dst.reg = tmp2;
setaddrmode_PVSSRC(&(pAsm->S[0].src), ADDR_ABSOLUTE);
pAsm->S[0].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[0].src.reg = tmp1;
noswizzle_PVSSRC(&(pAsm->S[0].src));
setaddrmode_PVSSRC(&(pAsm->S[1].src), ADDR_ABSOLUTE);
pAsm->S[1].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[1].src.reg = tmp1;
setswizzle_PVSSRC(&(pAsm->S[1].src), SQ_SEL_Z);
setaddrmode_PVSSRC(&(pAsm->S[2].src), ADDR_ABSOLUTE);
pAsm->S[2].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[2].src.reg = tmp1;
setswizzle_PVSSRC(&(pAsm->S[2].src), SQ_SEL_1);
next_ins(pAsm);
/* ADD the remaining .5 */
pAsm->D.dst.opcode = SQ_OP2_INST_ADD;
setaddrmode_PVSDST(&(pAsm->D.dst), ADDR_ABSOLUTE);
pAsm->D.dst.rtype = DST_REG_TEMPORARY;
pAsm->D.dst.reg = tmp2;
pAsm->D.dst.writex = 1;
pAsm->D.dst.writey = 1;
pAsm->D.dst.writez = 0;
pAsm->D.dst.writew = 0;
setaddrmode_PVSSRC(&(pAsm->S[0].src), ADDR_ABSOLUTE);
pAsm->S[0].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[0].src.reg = tmp2;
noswizzle_PVSSRC(&(pAsm->S[0].src));
setaddrmode_PVSSRC(&(pAsm->S[1].src), ADDR_ABSOLUTE);
pAsm->S[1].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[1].src.reg = 252; // SQ_ALU_SRC_0_5
noswizzle_PVSSRC(&(pAsm->S[1].src));
next_ins(pAsm);
/* tmp1.xy = temp2.xy */
pAsm->D.dst.opcode = SQ_OP2_INST_MOV;
setaddrmode_PVSDST(&(pAsm->D.dst), ADDR_ABSOLUTE);
pAsm->D.dst.rtype = DST_REG_TEMPORARY;
pAsm->D.dst.reg = tmp1;
pAsm->D.dst.writex = 1;
pAsm->D.dst.writey = 1;
pAsm->D.dst.writez = 0;
pAsm->D.dst.writew = 0;
setaddrmode_PVSSRC(&(pAsm->S[0].src), ADDR_ABSOLUTE);
pAsm->S[0].src.rtype = SRC_REG_TEMPORARY;
pAsm->S[0].src.reg = tmp2;
noswizzle_PVSSRC(&(pAsm->S[0].src));
next_ins(pAsm);
pAsm->aArgSubst[1] = tmp1;
need_barrier = GL_TRUE;
}
pAsm->D.dst.opcode = SQ_TEX_INST_SAMPLE;
pAsm->is_tex = GL_TRUE;
if ( GL_TRUE == need_barrier )
{
pAsm->need_tex_barrier = GL_TRUE;
}
// Set src1 to tex unit id // Set src1 to tex unit id
pAsm->S[1].src.reg = pAsm->pILInst[pAsm->uiCurInst].TexSrcUnit; pAsm->S[1].src.reg = pAsm->pILInst[pAsm->uiCurInst].TexSrcUnit;
pAsm->S[1].src.rtype = SRC_REG_TEMPORARY; pAsm->S[1].src.rtype = SRC_REG_TEMPORARY;
@@ -3440,11 +3637,26 @@ GLboolean assemble_TEX(r700_AssemblerBase *pAsm)
return GL_FALSE; return GL_FALSE;
} }
if ( GL_FALSE == next_ins(pAsm) ) if(pAsm->pILInst[pAsm->uiCurInst].Opcode == OPCODE_TXP)
{ {
return GL_FALSE; /* hopefully did swizzles before */
noswizzle_PVSSRC(&(pAsm->S[0].src));
} }
if(pAsm->pILInst[pAsm->uiCurInst].TexSrcTarget == TEXTURE_CUBE_INDEX)
{
/* SAMPLE dst, tmp.yxwy, CUBE */
pAsm->S[0].src.swizzlex = SQ_SEL_Y;
pAsm->S[0].src.swizzley = SQ_SEL_X;
pAsm->S[0].src.swizzlez = SQ_SEL_W;
pAsm->S[0].src.swizzlew = SQ_SEL_Y;
}
if ( GL_FALSE == next_ins(pAsm) )
{
return GL_FALSE;
}
return GL_TRUE; return GL_TRUE;
} }

View File

@@ -374,6 +374,10 @@ typedef struct r700_AssemblerBase
struct prog_instruction * pILInst; struct prog_instruction * pILInst;
GLuint uiCurInst; GLuint uiCurInst;
GLboolean bR6xx; GLboolean bR6xx;
/* helper to decide which type of instruction to assemble */
GLboolean is_tex;
/* we inserted helper intructions and need barrier on next TEX ins */
GLboolean need_tex_barrier;
} r700_AssemblerBase; } r700_AssemblerBase;
//Internal use //Internal use