| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183 |
- using Ryujinx.Graphics.Shader.Decoders;
- using Ryujinx.Graphics.Shader.IntermediateRepresentation;
- using Ryujinx.Graphics.Shader.Translation;
- using static Ryujinx.Graphics.Shader.Instructions.InstEmitHelper;
- using static Ryujinx.Graphics.Shader.IntermediateRepresentation.OperandHelper;
- namespace Ryujinx.Graphics.Shader.Instructions
- {
- static partial class InstEmit
- {
- public static void Vmnmx(EmitterContext context)
- {
- InstVmnmx op = context.GetOp<InstVmnmx>();
- Operand srcA = InstEmitAluHelper.Extend(context, GetSrcReg(context, op.SrcA), op.ASelect);
- Operand srcC = GetSrcReg(context, op.SrcC);
- Operand srcB;
- if (op.BVideo)
- {
- srcB = InstEmitAluHelper.Extend(context, GetSrcReg(context, op.SrcB), op.BSelect);
- }
- else
- {
- int imm = op.Imm16;
- if ((op.BSelect & VectorSelect.S8B0) != 0)
- {
- imm = (imm << 16) >> 16;
- }
- srcB = Const(imm);
- }
- Operand res;
- bool resSigned;
- if ((op.ASelect & VectorSelect.S8B0) != (op.BSelect & VectorSelect.S8B0))
- {
- // Signedness is different, but for max, result will always fit a U32,
- // since one of the inputs can't be negative, and the result is the one
- // with highest value. For min, it will always fit on a S32, since
- // one of the input can't be greater than INT_MAX and we want the lowest value.
- resSigned = !op.Mn;
- res = op.Mn ? context.IMaximumU32(srcA, srcB) : context.IMinimumS32(srcA, srcB);
- if ((op.ASelect & VectorSelect.S8B0) != 0)
- {
- Operand isBGtIntMax = context.ICompareLess(srcB, Const(0));
- res = context.ConditionalSelect(isBGtIntMax, srcB, res);
- }
- else
- {
- Operand isAGtIntMax = context.ICompareLess(srcA, Const(0));
- res = context.ConditionalSelect(isAGtIntMax, srcA, res);
- }
- }
- else
- {
- // Ra and Rb have the same signedness, so doesn't matter which one we test.
- resSigned = (op.ASelect & VectorSelect.S8B0) != 0;
- if (op.Mn)
- {
- res = resSigned
- ? context.IMaximumS32(srcA, srcB)
- : context.IMaximumU32(srcA, srcB);
- }
- else
- {
- res = resSigned
- ? context.IMinimumS32(srcA, srcB)
- : context.IMinimumU32(srcA, srcB);
- }
- }
- if (op.Sat)
- {
- if (op.DFormat && !resSigned)
- {
- res = context.IMinimumU32(res, Const(int.MaxValue));
- }
- else if (!op.DFormat && resSigned)
- {
- res = context.IMaximumS32(res, Const(0));
- }
- }
- switch (op.VideoOp)
- {
- case VideoOp.Acc:
- res = context.IAdd(res, srcC);
- break;
- case VideoOp.Max:
- res = op.DFormat ? context.IMaximumS32(res, srcC) : context.IMaximumU32(res, srcC);
- break;
- case VideoOp.Min:
- res = op.DFormat ? context.IMinimumS32(res, srcC) : context.IMinimumU32(res, srcC);
- break;
- case VideoOp.Mrg16h:
- res = context.BitfieldInsert(srcC, res, Const(16), Const(16));
- break;
- case VideoOp.Mrg16l:
- res = context.BitfieldInsert(srcC, res, Const(0), Const(16));
- break;
- case VideoOp.Mrg8b0:
- res = context.BitfieldInsert(srcC, res, Const(0), Const(8));
- break;
- case VideoOp.Mrg8b2:
- res = context.BitfieldInsert(srcC, res, Const(16), Const(8));
- break;
- }
- context.Copy(GetDest(op.Dest), res);
- }
- public static void Vsetp(EmitterContext context)
- {
- InstVsetp op = context.GetOp<InstVsetp>();
- Operand srcA = InstEmitAluHelper.Extend(context, GetSrcReg(context, op.SrcA), op.ASelect);
- Operand srcB;
- if (op.BVideo)
- {
- srcB = InstEmitAluHelper.Extend(context, GetSrcReg(context, op.SrcB), op.BSelect);
- }
- else
- {
- int imm = op.Imm16;
- if ((op.BSelect & VectorSelect.S8B0) != 0)
- {
- imm = (imm << 16) >> 16;
- }
- srcB = Const(imm);
- }
- Operand p0Res;
- bool signedA = (op.ASelect & VectorSelect.S8B0) != 0;
- bool signedB = (op.BSelect & VectorSelect.S8B0) != 0;
- if (signedA != signedB)
- {
- bool a32 = (op.ASelect & ~VectorSelect.S8B0) == VectorSelect.U32;
- bool b32 = (op.BSelect & ~VectorSelect.S8B0) == VectorSelect.U32;
- if (!a32 && !b32)
- {
- // Both values are extended small integer and can always fit in a S32, just do a signed comparison.
- p0Res = GetIntComparison(context, op.VComp, srcA, srcB, isSigned: true, extended: false);
- }
- else
- {
- // TODO: Mismatching sign case.
- p0Res = Const(0);
- }
- }
- else
- {
- // Sign matches, just do a regular comparison.
- p0Res = GetIntComparison(context, op.VComp, srcA, srcB, signedA, extended: false);
- }
- Operand p1Res = context.BitwiseNot(p0Res);
- Operand pred = GetPredicate(context, op.SrcPred, op.SrcPredInv);
- p0Res = InstEmitAluHelper.GetPredLogicalOp(context, op.BoolOp, p0Res, pred);
- p1Res = InstEmitAluHelper.GetPredLogicalOp(context, op.BoolOp, p1Res, pred);
- context.Copy(Register(op.DestPred, RegisterType.Predicate), p0Res);
- context.Copy(Register(op.DestPredInv, RegisterType.Predicate), p1Res);
- }
- }
- }
|