1 /****************************************************************************
2  * Copyright (C) 2014-2018 Intel Corporation.   All Rights Reserved.
3  *
4  * Permission is hereby granted, free of charge, to any person obtaining a
5  * copy of this software and associated documentation files (the "Software"),
6  * to deal in the Software without restriction, including without limitation
7  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8  * and/or sell copies of the Software, and to permit persons to whom the
9  * Software is furnished to do so, subject to the following conditions:
10  *
11  * The above copyright notice and this permission notice (including the next
12  * paragraph) shall be included in all copies or substantial portions of the
13  * Software.
14  *
15  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
18  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21  * IN THE SOFTWARE.
22  *
23  * @file fetch_jit.cpp
24  *
25  * @brief Implementation of the fetch jitter
26  *
27  * Notes:
28  *
29  ******************************************************************************/
30 #include "jit_pch.hpp"
31 #include "builder_gfx_mem.h"
32 #include "jit_api.h"
33 #include "fetch_jit.h"
34 #include "gen_state_llvm.h"
35 #include "functionpasses/passes.h"
36 
37 //#define FETCH_DUMP_VERTEX 1
38 using namespace llvm;
39 using namespace SwrJit;
40 
41 bool isComponentEnabled(ComponentEnable enableMask, uint8_t component);
42 
43 enum ConversionType
44 {
45     CONVERT_NONE,
46     CONVERT_NORMALIZED,
47     CONVERT_USCALED,
48     CONVERT_SSCALED,
49     CONVERT_SFIXED,
50 };
51 
52 //////////////////////////////////////////////////////////////////////////
53 /// Interface to Jitting a fetch shader
54 //////////////////////////////////////////////////////////////////////////
55 struct FetchJit : public BuilderGfxMem
56 {
FetchJitFetchJit57     FetchJit(JitManager* pJitMgr) : BuilderGfxMem(pJitMgr) {}
58 
59     Function* Create(const FETCH_COMPILE_STATE& fetchState);
60 
61     Value* GetSimdValid32bitIndices(Value* vIndices, Value* pLastIndex);
62     Value* GetSimdValid16bitIndices(Value* vIndices, Value* pLastIndex);
63     Value* GetSimdValid8bitIndices(Value* vIndices, Value* pLastIndex);
64     template <typename T>
65     Value* GetSimdValidIndicesHelper(Value* pIndices, Value* pLastIndex);
66 
67     // package up Shuffle*bpcGatherd args into a tuple for convenience
68     typedef std::tuple<Value*&,
69                        Value*,
70                        const Instruction::CastOps,
71                        const ConversionType,
72                        uint32_t&,
73                        uint32_t&,
74                        const ComponentEnable,
75                        const ComponentControl (&)[4],
76                        Value* (&)[4],
77                        const uint32_t (&)[4]>
78         Shuffle8bpcArgs;
79 
80     void Shuffle8bpcGatherd16(Shuffle8bpcArgs& args);
81     void Shuffle8bpcGatherd(Shuffle8bpcArgs& args);
82 
83     typedef std::tuple<Value* (&)[2],
84                        Value*,
85                        const Instruction::CastOps,
86                        const ConversionType,
87                        uint32_t&,
88                        uint32_t&,
89                        const ComponentEnable,
90                        const ComponentControl (&)[4],
91                        Value* (&)[4]>
92         Shuffle16bpcArgs;
93 
94     void Shuffle16bpcGather16(Shuffle16bpcArgs& args);
95     void Shuffle16bpcGather(Shuffle16bpcArgs& args);
96 
97     void StoreVertexElements(Value*         pVtxOut,
98                              const uint32_t outputElt,
99                              const uint32_t numEltsToStore,
100                              Value* (&vVertexElements)[4]);
101 
102     Value* GenerateCompCtrlVector(const ComponentControl ctrl);
103 
104     void JitGatherVertices(const FETCH_COMPILE_STATE& fetchState,
105                            Value*                     streams,
106                            Value*                     vIndices,
107                            Value*                     pVtxOut);
108 
109     bool IsOddFormat(SWR_FORMAT format);
110     bool IsUniformFormat(SWR_FORMAT format);
111     void UnpackComponents(SWR_FORMAT format, Value* vInput, Value* result[4]);
112     void CreateGatherOddFormats(
113         SWR_FORMAT format, Value* pMask, Value* pBase, Value* offsets, Value* result[4]);
114     void ConvertFormat(SWR_FORMAT format, Value* texels[4]);
115 
116     Value* mpFetchInfo;
117 };
118 
Create(const FETCH_COMPILE_STATE & fetchState)119 Function* FetchJit::Create(const FETCH_COMPILE_STATE& fetchState)
120 {
121     std::stringstream fnName("FCH_", std::ios_base::in | std::ios_base::out | std::ios_base::ate);
122     fnName << ComputeCRC(0, &fetchState, sizeof(fetchState));
123 
124     Function* fetch = Function::Create(
125         JM()->mFetchShaderTy, GlobalValue::ExternalLinkage, fnName.str(), JM()->mpCurrentModule);
126     BasicBlock* entry = BasicBlock::Create(JM()->mContext, "entry", fetch);
127 
128     fetch->getParent()->setModuleIdentifier(fetch->getName());
129 
130     IRB()->SetInsertPoint(entry);
131 
132     auto argitr = fetch->arg_begin();
133 
134     // Fetch shader arguments
135     Value* privateContext = &*argitr;
136     ++argitr;
137     privateContext->setName("privateContext");
138     SetPrivateContext(privateContext);
139 
140     mpWorkerData = &*argitr;
141     ++argitr;
142     mpWorkerData->setName("pWorkerData");
143 
144     mpFetchInfo = &*argitr;
145     ++argitr;
146     mpFetchInfo->setName("fetchInfo");
147     Value* pVtxOut = &*argitr;
148     pVtxOut->setName("vtxOutput");
149 
150     uint32_t baseWidth = mVWidth;
151 
152     SWR_ASSERT(mVWidth == 8 || mVWidth == 16, "Unsupported vector width %d", mVWidth);
153 
154     // Override builder target width to force 16-wide SIMD
155 #if USE_SIMD16_SHADERS
156     SetTargetWidth(16);
157 #endif
158 
159     pVtxOut = BITCAST(pVtxOut, PointerType::get(mSimdFP32Ty, 0));
160 
161     // SWR_FETCH_CONTEXT::pStreams
162     Value* streams = LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_pStreams});
163     streams->setName("pStreams");
164 
165     // SWR_FETCH_CONTEXT::pIndices
166     Value* indices = LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_xpIndices});
167     indices->setName("pIndices");
168 
169     // SWR_FETCH_CONTEXT::pLastIndex
170     Value* pLastIndex = LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_xpLastIndex});
171     pLastIndex->setName("pLastIndex");
172 
173     Value* vIndices;
174     switch (fetchState.indexType)
175     {
176     case R8_UINT:
177         indices = BITCAST(indices, Type::getInt8PtrTy(JM()->mContext, 0));
178         if (fetchState.bDisableIndexOOBCheck)
179         {
180             vIndices = LOAD(
181                 BITCAST(indices, PointerType::get(getVectorType(mInt8Ty, mpJitMgr->mVWidth), 0)),
182                 {(uint32_t)0});
183             vIndices = Z_EXT(vIndices, mSimdInt32Ty);
184         }
185         else
186         {
187             vIndices = GetSimdValid8bitIndices(indices, pLastIndex);
188         }
189         break;
190     case R16_UINT:
191         if (fetchState.bDisableIndexOOBCheck)
192         {
193             vIndices = LOAD(
194                 BITCAST(indices, PointerType::get(getVectorType(mInt16Ty, mpJitMgr->mVWidth), 0)),
195                 {(uint32_t)0});
196             vIndices = Z_EXT(vIndices, mSimdInt32Ty);
197         }
198         else
199         {
200             vIndices = GetSimdValid16bitIndices(indices, pLastIndex);
201         }
202         break;
203     case R32_UINT:
204         (fetchState.bDisableIndexOOBCheck)
205             ? vIndices = LOAD(indices,
206                               "",
207                               PointerType::get(mSimdInt32Ty, 0),
208                               MEM_CLIENT::GFX_MEM_CLIENT_FETCH)
209             : vIndices = GetSimdValid32bitIndices(indices, pLastIndex);
210         break; // incoming type is already 32bit int
211     default:
212         vIndices = nullptr;
213         assert(false && "Unsupported index type");
214         break;
215     }
216 
217     if (fetchState.bForceSequentialAccessEnable)
218     {
219         Value* pOffsets = mVWidth == 8 ? C({0, 1, 2, 3, 4, 5, 6, 7})
220                                        : C({0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15});
221 
222         // VertexData buffers are accessed sequentially, the index is equal to the vertex number
223         vIndices = VBROADCAST(LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_StartVertex}));
224         vIndices = ADD(vIndices, pOffsets);
225     }
226 
227     Value* vVertexId = vIndices;
228     if (fetchState.bVertexIDOffsetEnable)
229     {
230         // Assuming one of baseVertex or startVertex is 0, so adding both should be functionally
231         // correct
232         Value* vBaseVertex  = VBROADCAST(LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_BaseVertex}));
233         Value* vStartVertex = VBROADCAST(LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_StartVertex}));
234         vVertexId           = ADD(vIndices, vBaseVertex);
235         vVertexId           = ADD(vVertexId, vStartVertex);
236     }
237 
238     // store out vertex IDs
239     if (mVWidth == 16)
240     {
241         // store out in simd8 halves until core supports 16-wide natively
242         auto vVertexIdLo = EXTRACT_16(vVertexId, 0);
243         auto vVertexIdHi = EXTRACT_16(vVertexId, 1);
244         STORE(vVertexIdLo, GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_VertexID}));
245         STORE(vVertexIdHi, GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_VertexID2}));
246     }
247     else if (mVWidth == 8)
248     {
249         STORE(vVertexId, GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_VertexID}));
250     }
251 
252     // store out cut mask if enabled
253     if (fetchState.bEnableCutIndex)
254     {
255         Value* vCutIndex = VIMMED1(fetchState.cutIndex);
256         Value* cutMask   = VMASK(ICMP_EQ(vIndices, vCutIndex));
257 
258         if (mVWidth == 16)
259         {
260             auto cutMaskLo = EXTRACT_16(cutMask, 0);
261             auto cutMaskHi = EXTRACT_16(cutMask, 1);
262             STORE(cutMaskLo, GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_CutMask}));
263             STORE(cutMaskHi, GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_CutMask2}));
264         }
265         else if (mVWidth == 8)
266         {
267             STORE(cutMask, GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_CutMask}));
268         }
269     }
270 
271     // Fetch attributes from memory and output to a simdvertex struct
272     JitGatherVertices(fetchState, streams, vIndices, pVtxOut);
273 
274     RET_VOID();
275 
276     JitManager::DumpToFile(fetch, "src");
277 
278 #if defined(_DEBUG)
279     verifyFunction(*fetch);
280 #endif
281 
282     ::FunctionPassManager setupPasses(JM()->mpCurrentModule);
283 
284     ///@todo We don't need the CFG passes for fetch. (e.g. BreakCriticalEdges and CFGSimplification)
285     setupPasses.add(createBreakCriticalEdgesPass());
286     setupPasses.add(createCFGSimplificationPass());
287     setupPasses.add(createEarlyCSEPass());
288     setupPasses.add(createPromoteMemoryToRegisterPass());
289 
290     setupPasses.run(*fetch);
291 
292     JitManager::DumpToFile(fetch, "se");
293 
294     ::FunctionPassManager optPasses(JM()->mpCurrentModule);
295 
296     ///@todo Haven't touched these either. Need to remove some of these and add others.
297     optPasses.add(createCFGSimplificationPass());
298     optPasses.add(createEarlyCSEPass());
299     optPasses.add(createInstructionCombiningPass());
300 #if LLVM_VERSION_MAJOR <= 11
301     optPasses.add(createConstantPropagationPass());
302 #endif
303     optPasses.add(createSCCPPass());
304     optPasses.add(createAggressiveDCEPass());
305 
306     optPasses.run(*fetch);
307 
308     optPasses.add(createLowerX86Pass(this));
309     optPasses.run(*fetch);
310 
311     JitManager::DumpToFile(fetch, "opt");
312 
313 
314     // Revert 16-wide override
315 #if USE_SIMD16_SHADERS
316     SetTargetWidth(baseWidth);
317 #endif
318 
319     return fetch;
320 }
321 
322 // returns true for odd formats that require special state.gather handling
IsOddFormat(SWR_FORMAT format)323 bool FetchJit::IsOddFormat(SWR_FORMAT format)
324 {
325     const SWR_FORMAT_INFO& info = GetFormatInfo(format);
326     if (info.bpc[0] != 8 && info.bpc[0] != 16 && info.bpc[0] != 32 && info.bpc[0] != 64)
327     {
328         return true;
329     }
330     return false;
331 }
332 
333 // format is uniform if all components are the same size and type
IsUniformFormat(SWR_FORMAT format)334 bool FetchJit::IsUniformFormat(SWR_FORMAT format)
335 {
336     const SWR_FORMAT_INFO& info  = GetFormatInfo(format);
337     uint32_t               bpc0  = info.bpc[0];
338     uint32_t               type0 = info.type[0];
339 
340     for (uint32_t c = 1; c < info.numComps; ++c)
341     {
342         if (bpc0 != info.bpc[c] || type0 != info.type[c])
343         {
344             return false;
345         }
346     }
347     return true;
348 }
349 
350 // unpacks components based on format
351 // foreach component in the pixel
352 //   mask off everything but this component
353 //   shift component to LSB
UnpackComponents(SWR_FORMAT format,Value * vInput,Value * result[4])354 void FetchJit::UnpackComponents(SWR_FORMAT format, Value* vInput, Value* result[4])
355 {
356     const SWR_FORMAT_INFO& info = GetFormatInfo(format);
357 
358     uint32_t bitOffset = 0;
359     for (uint32_t c = 0; c < info.numComps; ++c)
360     {
361         uint32_t swizzledIndex = info.swizzle[c];
362         uint32_t compBits      = info.bpc[c];
363         uint32_t bitmask       = ((1 << compBits) - 1) << bitOffset;
364         Value*   comp          = AND(vInput, bitmask);
365         comp                   = LSHR(comp, bitOffset);
366 
367         result[swizzledIndex] = comp;
368         bitOffset += compBits;
369     }
370 }
371 
372 // gather for odd component size formats
373 // gather SIMD full pixels per lane then shift/mask to move each component to their
374 // own vector
CreateGatherOddFormats(SWR_FORMAT format,Value * pMask,Value * xpBase,Value * pOffsets,Value * pResult[4])375 void FetchJit::CreateGatherOddFormats(
376     SWR_FORMAT format, Value* pMask, Value* xpBase, Value* pOffsets, Value* pResult[4])
377 {
378     const SWR_FORMAT_INFO& info = GetFormatInfo(format);
379 
380     // only works if pixel size is <= 32bits
381     SWR_ASSERT(info.bpp <= 32);
382 
383     Value* pGather;
384     if (info.bpp == 32)
385     {
386         pGather =
387             GATHERDD(VIMMED1(0), xpBase, pOffsets, pMask, 1, MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
388     }
389     else
390     {
391         // Can't use 32-bit gather for items less than 32-bits, could cause page faults.
392         Value* pMem = ALLOCA(mSimdInt32Ty);
393         STORE(VIMMED1(0u), pMem);
394 
395         Value* pDstMem = POINTER_CAST(pMem, mInt32PtrTy);
396 
397         for (uint32_t lane = 0; lane < mVWidth; ++lane)
398         {
399             // Get index
400             Value* index = VEXTRACT(pOffsets, C(lane));
401             Value* mask  = VEXTRACT(pMask, C(lane));
402 
403             // use branch around load based on mask
404             // Needed to avoid page-faults on unmasked lanes
405             BasicBlock* pCurrentBB = IRB()->GetInsertBlock();
406             BasicBlock* pMaskedLoadBlock =
407                 BasicBlock::Create(JM()->mContext, "MaskedLaneLoad", pCurrentBB->getParent());
408             BasicBlock* pEndLoadBB =
409                 BasicBlock::Create(JM()->mContext, "AfterMaskedLoad", pCurrentBB->getParent());
410 
411             COND_BR(mask, pMaskedLoadBlock, pEndLoadBB);
412 
413             JM()->mBuilder.SetInsertPoint(pMaskedLoadBlock);
414 
415             switch (info.bpp)
416             {
417             case 8:
418             {
419                 Value* pDst  = BITCAST(GEP(pDstMem, C(lane)), PointerType::get(mInt8Ty, 0));
420                 Value* xpSrc = ADD(xpBase, Z_EXT(index, xpBase->getType()));
421                 STORE(LOAD(xpSrc, "", mInt8PtrTy, MEM_CLIENT::GFX_MEM_CLIENT_FETCH), pDst);
422                 break;
423             }
424 
425             case 16:
426             {
427                 Value* pDst  = BITCAST(GEP(pDstMem, C(lane)), PointerType::get(mInt16Ty, 0));
428                 Value* xpSrc = ADD(xpBase, Z_EXT(index, xpBase->getType()));
429                 STORE(LOAD(xpSrc, "", mInt16PtrTy, MEM_CLIENT::GFX_MEM_CLIENT_FETCH), pDst);
430                 break;
431             }
432             break;
433 
434             case 24:
435             {
436                 // First 16-bits of data
437                 Value* pDst  = BITCAST(GEP(pDstMem, C(lane)), PointerType::get(mInt16Ty, 0));
438                 Value* xpSrc = ADD(xpBase, Z_EXT(index, xpBase->getType()));
439                 STORE(LOAD(xpSrc, "", mInt16PtrTy, MEM_CLIENT::GFX_MEM_CLIENT_FETCH), pDst);
440 
441                 // Last 8-bits of data
442                 pDst  = BITCAST(GEP(pDst, C(1)), PointerType::get(mInt8Ty, 0));
443                 xpSrc = ADD(xpSrc, C((int64_t)2));
444                 STORE(LOAD(xpSrc, "", mInt8PtrTy, MEM_CLIENT::GFX_MEM_CLIENT_FETCH), pDst);
445                 break;
446             }
447 
448             default:
449                 SWR_INVALID("Shouldn't have BPP = %d now", info.bpp);
450                 break;
451             }
452 
453             BR(pEndLoadBB);
454             JM()->mBuilder.SetInsertPoint(pEndLoadBB);
455         }
456 
457         pGather = LOAD(pMem);
458     }
459 
460     for (uint32_t comp = 0; comp < 4; ++comp)
461     {
462         pResult[comp] = VIMMED1((int)info.defaults[comp]);
463     }
464 
465     UnpackComponents(format, pGather, pResult);
466 
467     // cast to fp32
468     pResult[0] = BITCAST(pResult[0], mSimdFP32Ty);
469     pResult[1] = BITCAST(pResult[1], mSimdFP32Ty);
470     pResult[2] = BITCAST(pResult[2], mSimdFP32Ty);
471     pResult[3] = BITCAST(pResult[3], mSimdFP32Ty);
472 }
473 
ConvertFormat(SWR_FORMAT format,Value * texels[4])474 void FetchJit::ConvertFormat(SWR_FORMAT format, Value* texels[4])
475 {
476     const SWR_FORMAT_INFO& info = GetFormatInfo(format);
477 
478     for (uint32_t c = 0; c < info.numComps; ++c)
479     {
480         uint32_t compIndex = info.swizzle[c];
481 
482         // skip any conversion on UNUSED components
483         if (info.type[c] == SWR_TYPE_UNUSED)
484         {
485             continue;
486         }
487 
488         if (info.isNormalized[c])
489         {
490             if (info.type[c] == SWR_TYPE_SNORM)
491             {
492                 /// @todo The most-negative value maps to -1.0f. e.g. the 5-bit value 10000 maps to
493                 /// -1.0f.
494 
495                 /// result = c * (1.0f / (2^(n-1) - 1);
496                 uint32_t n        = info.bpc[c];
497                 uint32_t pow2     = 1 << (n - 1);
498                 float    scale    = 1.0f / (float)(pow2 - 1);
499                 Value*   vScale   = VIMMED1(scale);
500                 texels[compIndex] = BITCAST(texels[compIndex], mSimdInt32Ty);
501                 texels[compIndex] = SI_TO_FP(texels[compIndex], mSimdFP32Ty);
502                 texels[compIndex] = FMUL(texels[compIndex], vScale);
503             }
504             else
505             {
506                 SWR_ASSERT(info.type[c] == SWR_TYPE_UNORM);
507 
508                 /// result = c * (1.0f / (2^n - 1))
509                 uint32_t n    = info.bpc[c];
510                 uint32_t pow2 = 1 << n;
511                 // special case 24bit unorm format, which requires a full divide to meet ULP
512                 // requirement
513                 if (n == 24)
514                 {
515                     float  scale      = (float)(pow2 - 1);
516                     Value* vScale     = VIMMED1(scale);
517                     texels[compIndex] = BITCAST(texels[compIndex], mSimdInt32Ty);
518                     texels[compIndex] = SI_TO_FP(texels[compIndex], mSimdFP32Ty);
519                     texels[compIndex] = FDIV(texels[compIndex], vScale);
520                 }
521                 else
522                 {
523                     float  scale      = 1.0f / (float)(pow2 - 1);
524                     Value* vScale     = VIMMED1(scale);
525                     texels[compIndex] = BITCAST(texels[compIndex], mSimdInt32Ty);
526                     texels[compIndex] = UI_TO_FP(texels[compIndex], mSimdFP32Ty);
527                     texels[compIndex] = FMUL(texels[compIndex], vScale);
528                 }
529             }
530             continue;
531         }
532     }
533 }
534 
535 //////////////////////////////////////////////////////////////////////////
536 /// @brief Loads attributes from memory using AVX2 GATHER(s)
537 /// @param fetchState - info about attributes to be fetched from memory
538 /// @param streams - value pointer to the current vertex stream
539 /// @param vIndices - vector value of indices to gather
540 /// @param pVtxOut - value pointer to output simdvertex struct
JitGatherVertices(const FETCH_COMPILE_STATE & fetchState,Value * streams,Value * vIndices,Value * pVtxOut)541 void FetchJit::JitGatherVertices(const FETCH_COMPILE_STATE& fetchState,
542                                  Value*                     streams,
543                                  Value*                     vIndices,
544                                  Value*                     pVtxOut)
545 {
546     uint32_t currentVertexElement = 0;
547     uint32_t outputElt            = 0;
548     Value*   vVertexElements[4];
549 
550     Value* startVertex   = LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_StartVertex});
551     Value* startInstance = LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_StartInstance});
552     Value* curInstance   = LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_CurInstance});
553     Value* vBaseVertex   = VBROADCAST(LOAD(mpFetchInfo, {0, SWR_FETCH_CONTEXT_BaseVertex}));
554     curInstance->setName("curInstance");
555 
556     for (uint32_t nInputElt = 0; nInputElt < fetchState.numAttribs; nInputElt += 1)
557     {
558         const INPUT_ELEMENT_DESC& ied = fetchState.layout[nInputElt];
559 
560         // skip element if all components are disabled
561         if (ied.ComponentPacking == ComponentEnable::NONE)
562         {
563             continue;
564         }
565 
566         const SWR_FORMAT_INFO& info = GetFormatInfo((SWR_FORMAT)ied.Format);
567         SWR_ASSERT((info.bpp != 0), "Unsupported format in JitGatherVertices.");
568         uint32_t bpc =
569             info.bpp /
570             info.numComps; ///@todo Code below assumes all components are same size. Need to fix.
571 
572         Value* stream = LOAD(streams, {ied.StreamIndex, SWR_VERTEX_BUFFER_STATE_xpData});
573 
574         Value* stride  = LOAD(streams, {ied.StreamIndex, SWR_VERTEX_BUFFER_STATE_pitch});
575         Value* vStride = VBROADCAST(stride);
576 
577         // max vertex index that is fully in bounds
578         Value* maxVertex = GEP(streams, {C(ied.StreamIndex), C(SWR_VERTEX_BUFFER_STATE_maxVertex)});
579         maxVertex        = LOAD(maxVertex);
580 
581         Value* minVertex = NULL;
582         if (fetchState.bPartialVertexBuffer)
583         {
584             // min vertex index for low bounds OOB checking
585             minVertex = GEP(streams, {C(ied.StreamIndex), C(SWR_VERTEX_BUFFER_STATE_minVertex)});
586             minVertex = LOAD(minVertex);
587         }
588 
589         if (fetchState.bInstanceIDOffsetEnable)
590         {
591             // the InstanceID (curInstance) value is offset by StartInstanceLocation
592             curInstance = ADD(curInstance, startInstance);
593         }
594 
595         Value* vCurIndices;
596         Value* startOffset;
597         Value* vInstanceStride = VIMMED1(0);
598 
599         if (ied.InstanceEnable)
600         {
601             Value* stepRate = C(ied.InstanceAdvancementState);
602 
603             // prevent a div by 0 for 0 step rate
604             Value* isNonZeroStep = ICMP_UGT(stepRate, C(0));
605             stepRate             = SELECT(isNonZeroStep, stepRate, C(1));
606 
607             // calc the current offset into instanced data buffer
608             Value* calcInstance = UDIV(curInstance, stepRate);
609 
610             // if step rate is 0, every instance gets instance 0
611             calcInstance = SELECT(isNonZeroStep, calcInstance, C(0));
612 
613             vCurIndices = VBROADCAST(calcInstance);
614             startOffset = startInstance;
615         }
616         else if (ied.InstanceStrideEnable)
617         {
618             // grab the instance advancement state, determines stride in bytes from one instance to
619             // the next
620             Value* stepRate = C(ied.InstanceAdvancementState);
621             vInstanceStride = VBROADCAST(MUL(curInstance, stepRate));
622 
623             // offset indices by baseVertex
624             vCurIndices = ADD(vIndices, vBaseVertex);
625 
626             startOffset = startVertex;
627             SWR_ASSERT((0), "TODO: Fill out more once driver sends this down.");
628         }
629         else
630         {
631             // offset indices by baseVertex
632             vCurIndices = ADD(vIndices, vBaseVertex);
633             startOffset = startVertex;
634         }
635 
636         // All of the OOB calculations are in vertices, not VB offsets, to prevent having to
637         // do 64bit address offset calculations.
638 
639         // calculate byte offset to the start of the VB
640         Value* baseOffset = MUL(Z_EXT(startOffset, mInt64Ty), Z_EXT(stride, mInt64Ty));
641 
642         // VGATHER* takes an *i8 src pointer so that's what stream is
643         Value* pStreamBaseGFX = ADD(stream, baseOffset);
644 
645         // if we have a start offset, subtract from max vertex. Used for OOB check
646         maxVertex     = SUB(Z_EXT(maxVertex, mInt64Ty), Z_EXT(startOffset, mInt64Ty));
647         Value* maxNeg = ICMP_SLT(maxVertex, C((int64_t)0));
648         // if we have a negative value, we're already OOB. clamp at 0.
649         maxVertex = SELECT(maxNeg, C(0), TRUNC(maxVertex, mInt32Ty));
650 
651         if (fetchState.bPartialVertexBuffer)
652         {
653             // similary for min vertex
654             minVertex     = SUB(Z_EXT(minVertex, mInt64Ty), Z_EXT(startOffset, mInt64Ty));
655             Value* minNeg = ICMP_SLT(minVertex, C((int64_t)0));
656             minVertex     = SELECT(minNeg, C(0), TRUNC(minVertex, mInt32Ty));
657         }
658 
659         // Load the in bounds size of a partially valid vertex
660         Value* partialInboundsSize =
661             GEP(streams, {C(ied.StreamIndex), C(SWR_VERTEX_BUFFER_STATE_partialInboundsSize)});
662         partialInboundsSize       = LOAD(partialInboundsSize);
663         Value* vPartialVertexSize = VBROADCAST(partialInboundsSize);
664         Value* vBpp               = VBROADCAST(C(info.Bpp));
665         Value* vAlignmentOffsets  = VBROADCAST(C(ied.AlignedByteOffset));
666 
667         // is the element is <= the partially valid size
668         Value* vElementInBoundsMask = ICMP_SLE(vBpp, SUB(vPartialVertexSize, vAlignmentOffsets));
669 
670         // override cur indices with 0 if pitch is 0
671         Value* pZeroPitchMask = ICMP_EQ(vStride, VIMMED1(0));
672         vCurIndices           = SELECT(pZeroPitchMask, VIMMED1(0), vCurIndices);
673 
674         // are vertices partially OOB?
675         Value* vMaxVertex      = VBROADCAST(maxVertex);
676         Value* vPartialOOBMask = ICMP_EQ(vCurIndices, vMaxVertex);
677 
678         // are vertices fully in bounds?
679         Value* vMaxGatherMask = ICMP_ULT(vCurIndices, vMaxVertex);
680 
681         Value* vGatherMask;
682         if (fetchState.bPartialVertexBuffer)
683         {
684             // are vertices below minVertex limit?
685             Value* vMinVertex     = VBROADCAST(minVertex);
686             Value* vMinGatherMask = ICMP_UGE(vCurIndices, vMinVertex);
687 
688             // only fetch lanes that pass both tests
689             vGatherMask = AND(vMaxGatherMask, vMinGatherMask);
690         }
691         else
692         {
693             vGatherMask = vMaxGatherMask;
694         }
695 
696         // blend in any partially OOB indices that have valid elements
697         vGatherMask = SELECT(vPartialOOBMask, vElementInBoundsMask, vGatherMask);
698 
699         // calculate the actual offsets into the VB
700         Value* vOffsets = MUL(vCurIndices, vStride);
701         vOffsets        = ADD(vOffsets, vAlignmentOffsets);
702 
703         // if instance stride enable is:
704         //  true  - add product of the instanceID and advancement state to the offst into the VB
705         //  false - value of vInstanceStride has been initialialized to zero
706         vOffsets = ADD(vOffsets, vInstanceStride);
707 
708         // Packing and component control
709         ComponentEnable        compMask = (ComponentEnable)ied.ComponentPacking;
710         const ComponentControl compCtrl[4]{(ComponentControl)ied.ComponentControl0,
711                                            (ComponentControl)ied.ComponentControl1,
712                                            (ComponentControl)ied.ComponentControl2,
713                                            (ComponentControl)ied.ComponentControl3};
714 
715         // Special gather/conversion for formats without equal component sizes
716         if (IsOddFormat((SWR_FORMAT)ied.Format))
717         {
718             Value* pResults[4];
719             CreateGatherOddFormats(
720                 (SWR_FORMAT)ied.Format, vGatherMask, pStreamBaseGFX, vOffsets, pResults);
721             ConvertFormat((SWR_FORMAT)ied.Format, pResults);
722 
723             for (uint32_t c = 0; c < 4; c += 1)
724             {
725                 if (isComponentEnabled(compMask, c))
726                 {
727                     vVertexElements[currentVertexElement++] = pResults[c];
728                     if (currentVertexElement > 3)
729                     {
730                         StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
731                         // reset to the next vVertexElement to output
732                         currentVertexElement = 0;
733                     }
734                 }
735             }
736         }
737         else if (info.type[0] == SWR_TYPE_FLOAT)
738         {
739             ///@todo: support 64 bit vb accesses
740             Value* gatherSrc = VIMMED1(0.0f);
741 
742             SWR_ASSERT(IsUniformFormat((SWR_FORMAT)ied.Format),
743                        "Unsupported format for standard gather fetch.");
744 
745             // Gather components from memory to store in a simdvertex structure
746             switch (bpc)
747             {
748             case 16:
749             {
750                 Value* vGatherResult[2];
751 
752                 // if we have at least one component out of x or y to fetch
753                 if (isComponentEnabled(compMask, 0) || isComponentEnabled(compMask, 1))
754                 {
755                     vGatherResult[0] = GATHERPS(gatherSrc, pStreamBaseGFX, vOffsets, vGatherMask, 1, MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
756                     // e.g. result of first 8x32bit integer gather for 16bit components
757                     // 256i - 0    1    2    3    4    5    6    7
758                     //        xyxy xyxy xyxy xyxy xyxy xyxy xyxy xyxy
759                     //
760                 }
761 
762                 // if we have at least one component out of z or w to fetch
763                 if (isComponentEnabled(compMask, 2) || isComponentEnabled(compMask, 3))
764                 {
765                     // offset base to the next components(zw) in the vertex to gather
766                     pStreamBaseGFX = ADD(pStreamBaseGFX, C((int64_t)4));
767 
768                     vGatherResult[1] = GATHERPS(gatherSrc, pStreamBaseGFX, vOffsets, vGatherMask, 1, MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
769                     // e.g. result of second 8x32bit integer gather for 16bit components
770                     // 256i - 0    1    2    3    4    5    6    7
771                     //        zwzw zwzw zwzw zwzw zwzw zwzw zwzw zwzw
772                     //
773                 }
774 
775                 // if we have at least one component to shuffle into place
776                 if (compMask)
777                 {
778                     Shuffle16bpcArgs args = std::forward_as_tuple(vGatherResult,
779                                                                   pVtxOut,
780                                                                   Instruction::CastOps::FPExt,
781                                                                   CONVERT_NONE,
782                                                                   currentVertexElement,
783                                                                   outputElt,
784                                                                   compMask,
785                                                                   compCtrl,
786                                                                   vVertexElements);
787 
788                     // Shuffle gathered components into place in simdvertex struct
789                     mVWidth == 16 ? Shuffle16bpcGather16(args)
790                                   : Shuffle16bpcGather(args); // outputs to vVertexElements ref
791                 }
792             }
793             break;
794             case 32:
795             {
796                 for (uint32_t i = 0; i < 4; i += 1)
797                 {
798                     if (isComponentEnabled(compMask, i))
799                     {
800                         // if we need to gather the component
801                         if (compCtrl[i] == StoreSrc)
802                         {
803                             // Gather a SIMD of vertices
804                             // APIs allow a 4GB range for offsets
805                             // However, GATHERPS uses signed 32-bit offsets, so +/- 2GB range :(
806                             // Add 2GB to the base pointer and 2GB to the offsets.  This makes
807                             // "negative" (large) offsets into positive offsets and small offsets
808                             // into negative offsets.
809                             Value* vNewOffsets = ADD(vOffsets, VIMMED1(0x80000000));
810                             vVertexElements[currentVertexElement++] =
811                                 GATHERPS(gatherSrc,
812                                          ADD(pStreamBaseGFX, C((uintptr_t)0x80000000U)),
813                                          vNewOffsets,
814                                          vGatherMask,
815                                          1,
816                                          MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
817                         }
818                         else
819                         {
820                             vVertexElements[currentVertexElement++] =
821                                 GenerateCompCtrlVector(compCtrl[i]);
822                         }
823 
824                         if (currentVertexElement > 3)
825                         {
826                             StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
827                             // reset to the next vVertexElement to output
828                             currentVertexElement = 0;
829                         }
830                     }
831 
832                     // offset base to the next component in the vertex to gather
833                     pStreamBaseGFX = ADD(pStreamBaseGFX, C((int64_t)4));
834                 }
835             }
836             break;
837             case 64:
838             {
839                 for (uint32_t i = 0; i < 4; i += 1)
840                 {
841                     if (isComponentEnabled(compMask, i))
842                     {
843                         // if we need to gather the component
844                         if (compCtrl[i] == StoreSrc)
845                         {
846                             Value* vShufLo;
847                             Value* vShufHi;
848                             Value* vShufAll;
849 
850                             if (mVWidth == 8)
851                             {
852                                 vShufLo  = C({0, 1, 2, 3});
853                                 vShufHi  = C({4, 5, 6, 7});
854                                 vShufAll = C({0, 1, 2, 3, 4, 5, 6, 7});
855                             }
856                             else
857                             {
858                                 SWR_ASSERT(mVWidth == 16);
859                                 vShufLo = C({0, 1, 2, 3, 4, 5, 6, 7});
860                                 vShufHi = C({8, 9, 10, 11, 12, 13, 14, 15});
861                                 vShufAll =
862                                     C({0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15});
863                             }
864 
865                             Value* vMaskLo = VSHUFFLE(vGatherMask, vGatherMask, vShufLo);
866                             Value* vMaskHi = VSHUFFLE(vGatherMask, vGatherMask, vShufHi);
867 
868                             Value* vOffsetsLo = VSHUFFLE(vOffsets, vOffsets, vShufLo);
869                             Value* vOffsetsHi = VSHUFFLE(vOffsets, vOffsets, vShufHi);
870 
871                             Value* vZeroDouble = VECTOR_SPLAT(
872                                 mVWidth / 2, ConstantFP::get(IRB()->getDoubleTy(), 0.0f));
873 
874                             Value* pGatherLo =
875                                 GATHERPD(vZeroDouble, pStreamBaseGFX, vOffsetsLo, vMaskLo);
876                             Value* pGatherHi =
877                                 GATHERPD(vZeroDouble, pStreamBaseGFX, vOffsetsHi, vMaskHi);
878 
879                             Value* pGather = VSHUFFLE(pGatherLo, pGatherHi, vShufAll);
880                             pGather        = FP_TRUNC(pGather, mSimdFP32Ty);
881 
882                             vVertexElements[currentVertexElement++] = pGather;
883                         }
884                         else
885                         {
886                             vVertexElements[currentVertexElement++] =
887                                 GenerateCompCtrlVector(compCtrl[i]);
888                         }
889 
890                         if (currentVertexElement > 3)
891                         {
892                             StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
893                             // reset to the next vVertexElement to output
894                             currentVertexElement = 0;
895                         }
896                     }
897 
898                     // offset base to the next component  in the vertex to gather
899                     pStreamBaseGFX = ADD(pStreamBaseGFX, C((int64_t)8));
900                 }
901             }
902             break;
903             default:
904                 SWR_INVALID("Tried to fetch invalid FP format");
905                 break;
906             }
907         }
908         else
909         {
910             Instruction::CastOps extendCastType = Instruction::CastOps::CastOpsEnd;
911             ConversionType       conversionType = CONVERT_NONE;
912 
913             SWR_ASSERT(IsUniformFormat((SWR_FORMAT)ied.Format),
914                        "Unsupported format for standard gather fetch.");
915 
916             switch (info.type[0])
917             {
918             case SWR_TYPE_UNORM:
919                 conversionType = CONVERT_NORMALIZED;
920             case SWR_TYPE_UINT:
921                 extendCastType = Instruction::CastOps::ZExt;
922                 break;
923             case SWR_TYPE_SNORM:
924                 conversionType = CONVERT_NORMALIZED;
925             case SWR_TYPE_SINT:
926                 extendCastType = Instruction::CastOps::SExt;
927                 break;
928             case SWR_TYPE_USCALED:
929                 conversionType = CONVERT_USCALED;
930                 extendCastType = Instruction::CastOps::UIToFP;
931                 break;
932             case SWR_TYPE_SSCALED:
933                 conversionType = CONVERT_SSCALED;
934                 extendCastType = Instruction::CastOps::SIToFP;
935                 break;
936             case SWR_TYPE_SFIXED:
937                 conversionType = CONVERT_SFIXED;
938                 extendCastType = Instruction::CastOps::SExt;
939                 break;
940             default:
941                 break;
942             }
943 
944             // value substituted when component of gather is masked
945             Value* gatherSrc = VIMMED1(0);
946 
947             // Gather components from memory to store in a simdvertex structure
948             switch (bpc)
949             {
950             case 8:
951             {
952                 // if we have at least one component to fetch
953                 if (compMask)
954                 {
955                     Value* vGatherResult = GATHERDD(gatherSrc,
956                                                     pStreamBaseGFX,
957                                                     vOffsets,
958                                                     vGatherMask,
959                                                     1,
960                                                     MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
961                     // e.g. result of an 8x32bit integer gather for 8bit components
962                     // 256i - 0    1    2    3    4    5    6    7
963                     //        xyzw xyzw xyzw xyzw xyzw xyzw xyzw xyzw
964 
965                     Shuffle8bpcArgs args = std::forward_as_tuple(vGatherResult,
966                                                                  pVtxOut,
967                                                                  extendCastType,
968                                                                  conversionType,
969                                                                  currentVertexElement,
970                                                                  outputElt,
971                                                                  compMask,
972                                                                  compCtrl,
973                                                                  vVertexElements,
974                                                                  info.swizzle);
975 
976                     // Shuffle gathered components into place in simdvertex struct
977                     mVWidth == 16 ? Shuffle8bpcGatherd16(args)
978                                   : Shuffle8bpcGatherd(args); // outputs to vVertexElements ref
979                 }
980             }
981             break;
982             case 16:
983             {
984                 Value* vGatherResult[2];
985 
986                 // if we have at least one component out of x or y to fetch
987                 if (isComponentEnabled(compMask, 0) || isComponentEnabled(compMask, 1))
988                 {
989                     vGatherResult[0] = GATHERDD(gatherSrc,
990                                                 pStreamBaseGFX,
991                                                 vOffsets,
992                                                 vGatherMask,
993                                                 1,
994                                                 MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
995                     // e.g. result of first 8x32bit integer gather for 16bit components
996                     // 256i - 0    1    2    3    4    5    6    7
997                     //        xyxy xyxy xyxy xyxy xyxy xyxy xyxy xyxy
998                     //
999                 }
1000 
1001                 // if we have at least one component out of z or w to fetch
1002                 if (isComponentEnabled(compMask, 2) || isComponentEnabled(compMask, 3))
1003                 {
1004                     // offset base to the next components(zw) in the vertex to gather
1005                     pStreamBaseGFX = ADD(pStreamBaseGFX, C((int64_t)4));
1006 
1007                     vGatherResult[1] = GATHERDD(gatherSrc,
1008                                                 pStreamBaseGFX,
1009                                                 vOffsets,
1010                                                 vGatherMask,
1011                                                 1,
1012                                                 MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
1013                     // e.g. result of second 8x32bit integer gather for 16bit components
1014                     // 256i - 0    1    2    3    4    5    6    7
1015                     //        zwzw zwzw zwzw zwzw zwzw zwzw zwzw zwzw
1016                     //
1017                 }
1018 
1019                 // if we have at least one component to shuffle into place
1020                 if (compMask)
1021                 {
1022                     Shuffle16bpcArgs args = std::forward_as_tuple(vGatherResult,
1023                                                                   pVtxOut,
1024                                                                   extendCastType,
1025                                                                   conversionType,
1026                                                                   currentVertexElement,
1027                                                                   outputElt,
1028                                                                   compMask,
1029                                                                   compCtrl,
1030                                                                   vVertexElements);
1031 
1032                     // Shuffle gathered components into place in simdvertex struct
1033                     mVWidth == 16 ? Shuffle16bpcGather16(args)
1034                                   : Shuffle16bpcGather(args); // outputs to vVertexElements ref
1035                 }
1036             }
1037             break;
1038             case 32:
1039             {
1040                 // Gathered components into place in simdvertex struct
1041                 for (uint32_t i = 0; i < 4; i++)
1042                 {
1043                     if (isComponentEnabled(compMask, i))
1044                     {
1045                         // if we need to gather the component
1046                         if (compCtrl[i] == StoreSrc)
1047                         {
1048                             Value* pGather = GATHERDD(gatherSrc,
1049                                                       pStreamBaseGFX,
1050                                                       vOffsets,
1051                                                       vGatherMask,
1052                                                       1,
1053                                                       MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
1054 
1055                             if (conversionType == CONVERT_USCALED)
1056                             {
1057                                 pGather = UI_TO_FP(pGather, mSimdFP32Ty);
1058                             }
1059                             else if (conversionType == CONVERT_SSCALED)
1060                             {
1061                                 pGather = SI_TO_FP(pGather, mSimdFP32Ty);
1062                             }
1063                             else if (conversionType == CONVERT_SFIXED)
1064                             {
1065                                 pGather = FMUL(SI_TO_FP(pGather, mSimdFP32Ty),
1066                                                VBROADCAST(C(1 / 65536.0f)));
1067                             }
1068 
1069                             vVertexElements[currentVertexElement++] = pGather;
1070 
1071                             // e.g. result of a single 8x32bit integer gather for 32bit components
1072                             // 256i - 0    1    2    3    4    5    6    7
1073                             //        xxxx xxxx xxxx xxxx xxxx xxxx xxxx xxxx
1074                         }
1075                         else
1076                         {
1077                             vVertexElements[currentVertexElement++] =
1078                                 GenerateCompCtrlVector(compCtrl[i]);
1079                         }
1080 
1081                         if (currentVertexElement > 3)
1082                         {
1083                             StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
1084 
1085                             // reset to the next vVertexElement to output
1086                             currentVertexElement = 0;
1087                         }
1088                     }
1089 
1090                     // offset base to the next component  in the vertex to gather
1091                     pStreamBaseGFX = ADD(pStreamBaseGFX, C((int64_t)4));
1092                 }
1093             }
1094             break;
1095             }
1096         }
1097     }
1098 
1099     // if we have a partially filled vVertexElement struct, output it
1100     if (currentVertexElement > 0)
1101     {
1102         StoreVertexElements(pVtxOut, outputElt++, currentVertexElement, vVertexElements);
1103     }
1104 }
1105 
1106 
1107 typedef void* (*PFN_TRANSLATEGFXADDRESS_FUNC)(void* pdc, gfxptr_t va, bool* out_pbNullTileAccessed, void* pWorkerData);
1108 
1109 template <typename T>
GetSimdValidIndicesGfx(gfxptr_t indices,gfxptr_t lastIndex,uint32_t vWidth,PFN_TRANSLATEGFXADDRESS_FUNC pfnTranslate,void * pdc,uint32_t * outIndices,void * pWorkerData)1110 void GetSimdValidIndicesGfx(gfxptr_t                     indices,
1111                             gfxptr_t                     lastIndex,
1112                             uint32_t                     vWidth,
1113                             PFN_TRANSLATEGFXADDRESS_FUNC pfnTranslate,
1114                             void*                        pdc,
1115                             uint32_t*                    outIndices,
1116                             void*                        pWorkerData)
1117 {
1118     SWR_ASSERT(outIndices != nullptr);
1119 
1120     gfxptr_t indexPtr = indices;
1121     for (int64_t lane = 0; lane < vWidth; lane++)
1122     {
1123         uint32_t index = 0;
1124 
1125         if (indexPtr < lastIndex)
1126         {
1127             // translate indexPtr and load from it
1128             T* addr = (T*)pfnTranslate(pdc, indexPtr, nullptr, pWorkerData);
1129             SWR_ASSERT(addr != nullptr);
1130             index = *addr;
1131         }
1132 
1133         // index to 32 bits and insert into the correct simd lane
1134         outIndices[lane] = index;
1135 
1136         indexPtr += sizeof(T);
1137     }
1138 }
1139 
GetSimdValid8bitIndicesGfx(gfxptr_t indices,gfxptr_t lastIndex,uint32_t vWidth,PFN_TRANSLATEGFXADDRESS_FUNC pfnTranslate,void * pdc,uint32_t * outIndices,void * pWorkerData)1140 void GetSimdValid8bitIndicesGfx(gfxptr_t                     indices,
1141                                 gfxptr_t                     lastIndex,
1142                                 uint32_t                     vWidth,
1143                                 PFN_TRANSLATEGFXADDRESS_FUNC pfnTranslate,
1144                                 void*                        pdc,
1145                                 uint32_t*                    outIndices,
1146                                 void*                        pWorkerData)
1147 {
1148     GetSimdValidIndicesGfx<uint8_t>(indices, lastIndex, vWidth, pfnTranslate, pdc, outIndices, pWorkerData);
1149 }
1150 
GetSimdValid16bitIndicesGfx(gfxptr_t indices,gfxptr_t lastIndex,uint32_t vWidth,PFN_TRANSLATEGFXADDRESS_FUNC pfnTranslate,void * pdc,uint32_t * outIndices,void * pWorkerData)1151 void GetSimdValid16bitIndicesGfx(gfxptr_t                     indices,
1152                                  gfxptr_t                     lastIndex,
1153                                  uint32_t                     vWidth,
1154                                  PFN_TRANSLATEGFXADDRESS_FUNC pfnTranslate,
1155                                  void*                        pdc,
1156                                  uint32_t*                    outIndices,
1157                                  void*                        pWorkerData)
1158 {
1159     GetSimdValidIndicesGfx<uint16_t>(indices, lastIndex, vWidth, pfnTranslate, pdc, outIndices, pWorkerData);
1160 }
1161 
1162 
1163 template <typename T>
GetSimdValidIndicesHelper(Value * pIndices,Value * pLastIndex)1164 Value* FetchJit::GetSimdValidIndicesHelper(Value* pIndices, Value* pLastIndex)
1165 {
1166     SWR_ASSERT(pIndices->getType() == mInt64Ty && pLastIndex->getType() == mInt64Ty,
1167                "Function expects gfxptr_t for both input parameters.");
1168 
1169     Type* Ty = nullptr;
1170 
1171     static_assert(sizeof(T) == sizeof(uint16_t) || sizeof(T) == sizeof(uint8_t),
1172                   "Unsupported type for use with GetSimdValidIndicesHelper<T>");
1173     constexpr bool bSize = (sizeof(T) == sizeof(uint16_t));
1174     if (bSize)
1175     {
1176         Ty = mInt16PtrTy;
1177     }
1178     else if (sizeof(T) == sizeof(uint8_t))
1179     {
1180         Ty = mInt8PtrTy;
1181     }
1182     else
1183     {
1184         SWR_ASSERT(false, "This should never happen as per static_assert above.");
1185     }
1186 
1187     Value* vIndices = VUNDEF_I();
1188 
1189     {
1190         // store 0 index on stack to be used to conditionally load from if index address is OOB
1191         Value* pZeroIndex = ALLOCA(Ty->getPointerElementType());
1192         STORE(C((T)0), pZeroIndex);
1193 
1194         // Load a SIMD of index pointers
1195         for (int64_t lane = 0; lane < mVWidth; lane++)
1196         {
1197             // Calculate the address of the requested index
1198             Value* pIndex = GEP(pIndices, C(lane), Ty);
1199 
1200             pLastIndex = INT_TO_PTR(pLastIndex, Ty);
1201 
1202             // check if the address is less than the max index,
1203             Value* mask = ICMP_ULT(pIndex, pLastIndex);
1204 
1205             // if valid, load the index. if not, load 0 from the stack
1206             Value* pValid = SELECT(mask, pIndex, pZeroIndex);
1207             Value* index  = LOAD(pValid, "valid index", Ty, MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
1208 
1209             // zero extended index to 32 bits and insert into the correct simd lane
1210             index    = Z_EXT(index, mInt32Ty);
1211             vIndices = VINSERT(vIndices, index, lane);
1212         }
1213     }
1214 
1215     return vIndices;
1216 }
1217 
1218 //////////////////////////////////////////////////////////////////////////
1219 /// @brief Loads a simd of valid indices. OOB indices are set to 0
1220 /// *Note* have to do 8bit index checking in scalar until we have AVX-512
1221 /// support
1222 /// @param pIndices - pointer to 8 bit indices
1223 /// @param pLastIndex - pointer to last valid index
GetSimdValid8bitIndices(Value * pIndices,Value * pLastIndex)1224 Value* FetchJit::GetSimdValid8bitIndices(Value* pIndices, Value* pLastIndex)
1225 {
1226     return GetSimdValidIndicesHelper<uint8_t>(pIndices, pLastIndex);
1227 }
1228 
1229 //////////////////////////////////////////////////////////////////////////
1230 /// @brief Loads a simd of valid indices. OOB indices are set to 0
1231 /// *Note* have to do 16bit index checking in scalar until we have AVX-512
1232 /// support
1233 /// @param pIndices - pointer to 16 bit indices
1234 /// @param pLastIndex - pointer to last valid index
GetSimdValid16bitIndices(Value * pIndices,Value * pLastIndex)1235 Value* FetchJit::GetSimdValid16bitIndices(Value* pIndices, Value* pLastIndex)
1236 {
1237     return GetSimdValidIndicesHelper<uint16_t>(pIndices, pLastIndex);
1238 }
1239 
1240 //////////////////////////////////////////////////////////////////////////
1241 /// @brief Loads a simd of valid indices. OOB indices are set to 0
1242 /// @param pIndices - pointer to 32 bit indices
1243 /// @param pLastIndex - pointer to last valid index
GetSimdValid32bitIndices(Value * pIndices,Value * pLastIndex)1244 Value* FetchJit::GetSimdValid32bitIndices(Value* pIndices, Value* pLastIndex)
1245 {
1246     DataLayout dL(JM()->mpCurrentModule);
1247     Value*     iLastIndex = pLastIndex;
1248     Value*     iIndices   = pIndices;
1249 
1250     // get the number of indices left in the buffer (endPtr - curPtr) / sizeof(index)
1251     Value* numIndicesLeft = SUB(iLastIndex, iIndices);
1252     numIndicesLeft        = TRUNC(numIndicesLeft, mInt32Ty);
1253     numIndicesLeft        = SDIV(numIndicesLeft, C(4));
1254 
1255     // create a vector of index counts from the base index ptr passed into the fetch
1256     Constant* vIndexOffsets;
1257     if (mVWidth == 8)
1258     {
1259         vIndexOffsets = C({0, 1, 2, 3, 4, 5, 6, 7});
1260     }
1261     else
1262     {
1263         vIndexOffsets = C({0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15});
1264     }
1265 
1266     // compare index count to the max valid index
1267     // e.g vMaxIndex      4 4 4 4 4 4 4 4 : 4 indices left to load
1268     //     vIndexOffsets  0 1 2 3 4 5 6 7
1269     //     ------------------------------
1270     //     vIndexMask    -1-1-1-1 0 0 0 0 : offsets < max pass
1271     //     vLoadedIndices 0 1 2 3 0 0 0 0 : offsets >= max masked to 0
1272     Value* vMaxIndex  = VBROADCAST(numIndicesLeft);
1273     Value* vIndexMask = ICMP_SGT(vMaxIndex, vIndexOffsets);
1274 
1275     // Load the indices; OOB loads 0
1276     return MASKED_LOAD(pIndices,
1277                        4,
1278                        vIndexMask,
1279                        VIMMED1(0),
1280                        "vIndices",
1281                        PointerType::get(mSimdInt32Ty, 0),
1282                        MEM_CLIENT::GFX_MEM_CLIENT_FETCH);
1283 }
1284 
1285 //////////////////////////////////////////////////////////////////////////
1286 /// @brief Takes a SIMD of gathered 8bpc verts, zero or sign extends,
1287 /// denormalizes if needed, converts to F32 if needed, and positions in
1288 //  the proper SIMD rows to be output to the simdvertex structure
1289 /// @param args: (tuple of args, listed below)
1290 ///   @param vGatherResult - 8 gathered 8bpc vertices
1291 ///   @param pVtxOut - base pointer to output simdvertex struct
1292 ///   @param extendType - sign extend or zero extend
1293 ///   @param bNormalized - do we need to denormalize?
1294 ///   @param currentVertexElement - reference to the current vVertexElement
1295 ///   @param outputElt - reference to the current offset from simdvertex we're o
1296 ///   @param compMask - component packing mask
1297 ///   @param compCtrl - component control val
1298 ///   @param vVertexElements[4] - vertex components to output
1299 ///   @param swizzle[4] - component swizzle location
Shuffle8bpcGatherd16(Shuffle8bpcArgs & args)1300 void FetchJit::Shuffle8bpcGatherd16(Shuffle8bpcArgs& args)
1301 {
1302     // Unpack tuple args
1303     Value*&                    vGatherResult        = std::get<0>(args);
1304     Value*                     pVtxOut              = std::get<1>(args);
1305     const Instruction::CastOps extendType           = std::get<2>(args);
1306     const ConversionType       conversionType       = std::get<3>(args);
1307     uint32_t&                  currentVertexElement = std::get<4>(args);
1308     uint32_t&                  outputElt            = std::get<5>(args);
1309     const ComponentEnable      compMask             = std::get<6>(args);
1310     const ComponentControl(&compCtrl)[4]            = std::get<7>(args);
1311     Value*(&vVertexElements)[4]                     = std::get<8>(args);
1312     const uint32_t(&swizzle)[4]                     = std::get<9>(args);
1313 
1314     // cast types
1315     Type* vGatherTy = getVectorType(mInt32Ty, 8);
1316     Type* v32x8Ty   = getVectorType(mInt8Ty, 32);
1317 
1318     // have to do extra work for sign extending
1319     if ((extendType == Instruction::CastOps::SExt) || (extendType == Instruction::CastOps::SIToFP))
1320     {
1321         Type* v16x8Ty = getVectorType(mInt8Ty, 16); // 8x16bit ints in a 128bit lane
1322         Type* v128Ty  = getVectorType(IntegerType::getIntNTy(JM()->mContext, 128), 2);
1323 
1324         // shuffle mask, including any swizzling
1325         const char x          = (char)swizzle[0];
1326         const char y          = (char)swizzle[1];
1327         const char z          = (char)swizzle[2];
1328         const char w          = (char)swizzle[3];
1329         Value*     vConstMask = C<char>(
1330             {char(x),     char(x + 4),  char(x + 8), char(x + 12), char(y),     char(y + 4),
1331              char(y + 8), char(y + 12), char(z),     char(z + 4),  char(z + 8), char(z + 12),
1332              char(w),     char(w + 4),  char(w + 8), char(w + 12), char(x),     char(x + 4),
1333              char(x + 8), char(x + 12), char(y),     char(y + 4),  char(y + 8), char(y + 12),
1334              char(z),     char(z + 4),  char(z + 8), char(z + 12), char(w),     char(w + 4),
1335              char(w + 8), char(w + 12)});
1336 
1337         // SIMD16 PSHUFB isnt part of AVX-512F, so split into SIMD8 for the sake of KNL, for now..
1338 
1339         Value* vGatherResult_lo = EXTRACT_16(vGatherResult, 0);
1340         Value* vGatherResult_hi = EXTRACT_16(vGatherResult, 1);
1341 
1342         Value* vShufResult_lo =
1343             BITCAST(PSHUFB(BITCAST(vGatherResult_lo, v32x8Ty), vConstMask), vGatherTy);
1344         Value* vShufResult_hi =
1345             BITCAST(PSHUFB(BITCAST(vGatherResult_hi, v32x8Ty), vConstMask), vGatherTy);
1346 
1347         // after pshufb: group components together in each 128bit lane
1348         // 256i - 0    1    2    3    4    5    6    7
1349         //        xxxx yyyy zzzz wwww xxxx yyyy zzzz wwww
1350 
1351         Value* vi128XY_lo = nullptr;
1352         Value* vi128XY_hi = nullptr;
1353         if (isComponentEnabled(compMask, 0) || isComponentEnabled(compMask, 1))
1354         {
1355             vi128XY_lo = BITCAST(
1356                 VSHUFFLE(vShufResult_lo, vShufResult_lo, C<int32_t>({0, 4, 0, 0, 1, 5, 0, 0})),
1357                 v128Ty);
1358             vi128XY_hi = BITCAST(
1359                 VSHUFFLE(vShufResult_hi, vShufResult_hi, C<int32_t>({0, 4, 0, 0, 1, 5, 0, 0})),
1360                 v128Ty);
1361 
1362             // after PERMD: move and pack xy and zw components in low 64 bits of each 128bit lane
1363             // 256i - 0    1    2    3    4    5    6    7
1364             //        xxxx xxxx dcdc dcdc yyyy yyyy dcdc dcdc (dc - don't care)
1365         }
1366 
1367         // do the same for zw components
1368         Value* vi128ZW_lo = nullptr;
1369         Value* vi128ZW_hi = nullptr;
1370         if (isComponentEnabled(compMask, 2) || isComponentEnabled(compMask, 3))
1371         {
1372             vi128ZW_lo = BITCAST(
1373                 VSHUFFLE(vShufResult_lo, vShufResult_lo, C<int32_t>({2, 6, 0, 0, 3, 7, 0, 0})),
1374                 v128Ty);
1375             vi128ZW_hi = BITCAST(
1376                 VSHUFFLE(vShufResult_hi, vShufResult_hi, C<int32_t>({2, 6, 0, 0, 3, 7, 0, 0})),
1377                 v128Ty);
1378         }
1379 
1380         // init denormalize variables if needed
1381         Instruction::CastOps fpCast;
1382         Value*               conversionFactor;
1383 
1384         switch (conversionType)
1385         {
1386         case CONVERT_NORMALIZED:
1387             fpCast           = Instruction::CastOps::SIToFP;
1388             conversionFactor = VIMMED1((float)(1.0 / 127.0));
1389             break;
1390         case CONVERT_SSCALED:
1391             fpCast           = Instruction::CastOps::SIToFP;
1392             conversionFactor = VIMMED1((float)(1.0));
1393             break;
1394         case CONVERT_USCALED:
1395             assert(false && "Type should not be sign extended!");
1396             conversionFactor = nullptr;
1397             break;
1398         default:
1399             assert(conversionType == CONVERT_NONE);
1400             conversionFactor = nullptr;
1401             break;
1402         }
1403 
1404         // sign extend all enabled components. If we have a fill vVertexElements, output to current
1405         // simdvertex
1406         for (uint32_t i = 0; i < 4; i++)
1407         {
1408             if (isComponentEnabled(compMask, i))
1409             {
1410                 if (compCtrl[i] == ComponentControl::StoreSrc)
1411                 {
1412                     // if x or z, extract 128bits from lane 0, else for y or w, extract from lane 1
1413                     uint32_t lane = ((i == 0) || (i == 2)) ? 0 : 1;
1414                     // if x or y, use vi128XY permute result, else use vi128ZW
1415                     Value* selectedPermute_lo = (i < 2) ? vi128XY_lo : vi128ZW_lo;
1416                     Value* selectedPermute_hi = (i < 2) ? vi128XY_hi : vi128ZW_hi;
1417 
1418                     // sign extend
1419                     Value* temp_lo =
1420                         PMOVSXBD(BITCAST(VEXTRACT(selectedPermute_lo, C(lane)), v16x8Ty));
1421                     Value* temp_hi =
1422                         PMOVSXBD(BITCAST(VEXTRACT(selectedPermute_hi, C(lane)), v16x8Ty));
1423 
1424                     Value* temp = JOIN_16(temp_lo, temp_hi);
1425 
1426                     // denormalize if needed
1427                     if (conversionType != CONVERT_NONE)
1428                     {
1429                         temp = FMUL(CAST(fpCast, temp, mSimdFP32Ty), conversionFactor);
1430                     }
1431 
1432                     vVertexElements[currentVertexElement] = temp;
1433 
1434                     currentVertexElement += 1;
1435                 }
1436                 else
1437                 {
1438                     vVertexElements[currentVertexElement++] = GenerateCompCtrlVector(compCtrl[i]);
1439                 }
1440 
1441                 if (currentVertexElement > 3)
1442                 {
1443                     StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
1444                     // reset to the next vVertexElement to output
1445                     currentVertexElement = 0;
1446                 }
1447             }
1448         }
1449     }
1450     // else zero extend
1451     else if ((extendType == Instruction::CastOps::ZExt) ||
1452              (extendType == Instruction::CastOps::UIToFP))
1453     {
1454         // init denormalize variables if needed
1455         Instruction::CastOps fpCast;
1456         Value*               conversionFactor;
1457 
1458         switch (conversionType)
1459         {
1460         case CONVERT_NORMALIZED:
1461             fpCast           = Instruction::CastOps::UIToFP;
1462             conversionFactor = VIMMED1((float)(1.0 / 255.0));
1463             break;
1464         case CONVERT_USCALED:
1465             fpCast           = Instruction::CastOps::UIToFP;
1466             conversionFactor = VIMMED1((float)(1.0));
1467             break;
1468         case CONVERT_SSCALED:
1469             assert(false && "Type should not be zero extended!");
1470             conversionFactor = nullptr;
1471             break;
1472         default:
1473             assert(conversionType == CONVERT_NONE);
1474             conversionFactor = nullptr;
1475             break;
1476         }
1477 
1478         // shuffle enabled components into lower byte of each 32bit lane, 0 extending to 32 bits
1479         for (uint32_t i = 0; i < 4; i++)
1480         {
1481             if (isComponentEnabled(compMask, i))
1482             {
1483                 if (compCtrl[i] == ComponentControl::StoreSrc)
1484                 {
1485                     // pshufb masks for each component
1486                     Value* vConstMask;
1487                     switch (swizzle[i])
1488                     {
1489                     case 0:
1490                         // x shuffle mask
1491                         vConstMask =
1492                             C<char>({0, -1, -1, -1, 4, -1, -1, -1, 8, -1, -1, -1, 12, -1, -1, -1,
1493                                      0, -1, -1, -1, 4, -1, -1, -1, 8, -1, -1, -1, 12, -1, -1, -1});
1494                         break;
1495                     case 1:
1496                         // y shuffle mask
1497                         vConstMask =
1498                             C<char>({1, -1, -1, -1, 5, -1, -1, -1, 9, -1, -1, -1, 13, -1, -1, -1,
1499                                      1, -1, -1, -1, 5, -1, -1, -1, 9, -1, -1, -1, 13, -1, -1, -1});
1500                         break;
1501                     case 2:
1502                         // z shuffle mask
1503                         vConstMask =
1504                             C<char>({2, -1, -1, -1, 6, -1, -1, -1, 10, -1, -1, -1, 14, -1, -1, -1,
1505                                      2, -1, -1, -1, 6, -1, -1, -1, 10, -1, -1, -1, 14, -1, -1, -1});
1506                         break;
1507                     case 3:
1508                         // w shuffle mask
1509                         vConstMask =
1510                             C<char>({3, -1, -1, -1, 7, -1, -1, -1, 11, -1, -1, -1, 15, -1, -1, -1,
1511                                      3, -1, -1, -1, 7, -1, -1, -1, 11, -1, -1, -1, 15, -1, -1, -1});
1512                         break;
1513                     default:
1514                         assert(false && "Invalid component");
1515                         vConstMask = nullptr;
1516                         break;
1517                     }
1518 
1519                     Value* vGatherResult_lo = EXTRACT_16(vGatherResult, 0);
1520                     Value* vGatherResult_hi = EXTRACT_16(vGatherResult, 1);
1521 
1522                     Value* temp_lo =
1523                         BITCAST(PSHUFB(BITCAST(vGatherResult_lo, v32x8Ty), vConstMask), vGatherTy);
1524                     Value* temp_hi =
1525                         BITCAST(PSHUFB(BITCAST(vGatherResult_hi, v32x8Ty), vConstMask), vGatherTy);
1526 
1527                     // after pshufb for x channel
1528                     // 256i - 0    1    2    3    4    5    6    7
1529                     //        x000 x000 x000 x000 x000 x000 x000 x000
1530 
1531                     Value* temp = JOIN_16(temp_lo, temp_hi);
1532 
1533                     // denormalize if needed
1534                     if (conversionType != CONVERT_NONE)
1535                     {
1536                         temp = FMUL(CAST(fpCast, temp, mSimdFP32Ty), conversionFactor);
1537                     }
1538 
1539                     vVertexElements[currentVertexElement] = temp;
1540 
1541                     currentVertexElement += 1;
1542                 }
1543                 else
1544                 {
1545                     vVertexElements[currentVertexElement++] = GenerateCompCtrlVector(compCtrl[i]);
1546                 }
1547 
1548                 if (currentVertexElement > 3)
1549                 {
1550                     StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
1551                     // reset to the next vVertexElement to output
1552                     currentVertexElement = 0;
1553                 }
1554             }
1555         }
1556     }
1557     else
1558     {
1559         SWR_INVALID("Unsupported conversion type");
1560     }
1561 }
1562 
Shuffle8bpcGatherd(Shuffle8bpcArgs & args)1563 void FetchJit::Shuffle8bpcGatherd(Shuffle8bpcArgs& args)
1564 {
1565     // Unpack tuple args
1566     Value*&                    vGatherResult        = std::get<0>(args);
1567     Value*                     pVtxOut              = std::get<1>(args);
1568     const Instruction::CastOps extendType           = std::get<2>(args);
1569     const ConversionType       conversionType       = std::get<3>(args);
1570     uint32_t&                  currentVertexElement = std::get<4>(args);
1571     uint32_t&                  outputElt            = std::get<5>(args);
1572     const ComponentEnable      compMask             = std::get<6>(args);
1573     const ComponentControl(&compCtrl)[4]            = std::get<7>(args);
1574     Value*(&vVertexElements)[4]                     = std::get<8>(args);
1575     const uint32_t(&swizzle)[4]                     = std::get<9>(args);
1576 
1577     // cast types
1578     Type* v32x8Ty = getVectorType(mInt8Ty, mVWidth * 4); // vwidth is units of 32 bits
1579 
1580     for (uint32_t i = 0; i < 4; i++)
1581     {
1582         if (!isComponentEnabled(compMask, i))
1583             continue;
1584 
1585         if (compCtrl[i] == ComponentControl::StoreSrc)
1586         {
1587 #if LLVM_VERSION_MAJOR >= 11
1588             using MaskType = int32_t;
1589 #else
1590             using MaskType = uint32_t;
1591 #endif
1592             std::vector<MaskType> vShuffleMasks[4] = {
1593                 {0, 4, 8, 12, 16, 20, 24, 28},  // x
1594                 {1, 5, 9, 13, 17, 21, 25, 29},  // y
1595                 {2, 6, 10, 14, 18, 22, 26, 30}, // z
1596                 {3, 7, 11, 15, 19, 23, 27, 31}, // w
1597             };
1598 
1599             Value* val = VSHUFFLE(BITCAST(vGatherResult, v32x8Ty),
1600                                   UndefValue::get(v32x8Ty),
1601                                   vShuffleMasks[swizzle[i]]);
1602 
1603             if ((extendType == Instruction::CastOps::SExt) ||
1604                 (extendType == Instruction::CastOps::SIToFP))
1605             {
1606                 switch (conversionType)
1607                 {
1608                 case CONVERT_NORMALIZED:
1609                     val = FMUL(SI_TO_FP(val, mSimdFP32Ty), VIMMED1((float)(1.0 / 127.0)));
1610                     break;
1611                 case CONVERT_SSCALED:
1612                     val = SI_TO_FP(val, mSimdFP32Ty);
1613                     break;
1614                 case CONVERT_USCALED:
1615                     SWR_INVALID("Type should not be sign extended!");
1616                     break;
1617                 default:
1618                     SWR_ASSERT(conversionType == CONVERT_NONE);
1619                     val = S_EXT(val, mSimdInt32Ty);
1620                     break;
1621                 }
1622             }
1623             else if ((extendType == Instruction::CastOps::ZExt) ||
1624                      (extendType == Instruction::CastOps::UIToFP))
1625             {
1626                 switch (conversionType)
1627                 {
1628                 case CONVERT_NORMALIZED:
1629                     val = FMUL(UI_TO_FP(val, mSimdFP32Ty), VIMMED1((float)(1.0 / 255.0)));
1630                     break;
1631                 case CONVERT_SSCALED:
1632                     SWR_INVALID("Type should not be zero extended!");
1633                     break;
1634                 case CONVERT_USCALED:
1635                     val = UI_TO_FP(val, mSimdFP32Ty);
1636                     break;
1637                 default:
1638                     SWR_ASSERT(conversionType == CONVERT_NONE);
1639                     val = Z_EXT(val, mSimdInt32Ty);
1640                     break;
1641                 }
1642             }
1643             else
1644             {
1645                 SWR_INVALID("Unsupported conversion type");
1646             }
1647 
1648             vVertexElements[currentVertexElement++] = val;
1649         }
1650         else
1651         {
1652             vVertexElements[currentVertexElement++] = GenerateCompCtrlVector(compCtrl[i]);
1653         }
1654 
1655         if (currentVertexElement > 3)
1656         {
1657             StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
1658             // reset to the next vVertexElement to output
1659             currentVertexElement = 0;
1660         }
1661     }
1662 }
1663 
1664 //////////////////////////////////////////////////////////////////////////
1665 /// @brief Takes a SIMD of gathered 16bpc verts, zero or sign extends,
1666 /// denormalizes if needed, converts to F32 if needed, and positions in
1667 //  the proper SIMD rows to be output to the simdvertex structure
1668 /// @param args: (tuple of args, listed below)
1669 ///   @param vGatherResult[2] - array of gathered 16bpc vertices, 4 per index
1670 ///   @param pVtxOut - base pointer to output simdvertex struct
1671 ///   @param extendType - sign extend or zero extend
1672 ///   @param bNormalized - do we need to denormalize?
1673 ///   @param currentVertexElement - reference to the current vVertexElement
1674 ///   @param outputElt - reference to the current offset from simdvertex we're o
1675 ///   @param compMask - component packing mask
1676 ///   @param compCtrl - component control val
1677 ///   @param vVertexElements[4] - vertex components to output
Shuffle16bpcGather16(Shuffle16bpcArgs & args)1678 void FetchJit::Shuffle16bpcGather16(Shuffle16bpcArgs& args)
1679 {
1680     // Unpack tuple args
1681     Value*(&vGatherResult)[2]                       = std::get<0>(args);
1682     Value*                     pVtxOut              = std::get<1>(args);
1683     const Instruction::CastOps extendType           = std::get<2>(args);
1684     const ConversionType       conversionType       = std::get<3>(args);
1685     uint32_t&                  currentVertexElement = std::get<4>(args);
1686     uint32_t&                  outputElt            = std::get<5>(args);
1687     const ComponentEnable      compMask             = std::get<6>(args);
1688     const ComponentControl(&compCtrl)[4]            = std::get<7>(args);
1689     Value*(&vVertexElements)[4]                     = std::get<8>(args);
1690 
1691     // cast types
1692     Type* vGatherTy = getVectorType(mInt32Ty, 8);
1693     Type* v32x8Ty   = getVectorType(mInt8Ty, 32);
1694 
1695     // have to do extra work for sign extending
1696     if ((extendType == Instruction::CastOps::SExt) ||
1697         (extendType == Instruction::CastOps::SIToFP) || (extendType == Instruction::CastOps::FPExt))
1698     {
1699         // is this PP float?
1700         bool bFP = (extendType == Instruction::CastOps::FPExt) ? true : false;
1701 
1702         Type* v8x16Ty   = getVectorType(mInt16Ty, 8); // 8x16bit in a 128bit lane
1703         Type* v128bitTy = getVectorType(IntegerType::getIntNTy(JM()->mContext, 128), 2);
1704 
1705         // shuffle mask
1706         Value* vConstMask = C<uint8_t>({0, 1, 4, 5, 8, 9, 12, 13, 2, 3, 6, 7, 10, 11, 14, 15,
1707                                         0, 1, 4, 5, 8, 9, 12, 13, 2, 3, 6, 7, 10, 11, 14, 15});
1708         Value* vi128XY_lo = nullptr;
1709         Value* vi128XY_hi = nullptr;
1710         if (isComponentEnabled(compMask, 0) || isComponentEnabled(compMask, 1))
1711         {
1712             // SIMD16 PSHUFB isnt part of AVX-512F, so split into SIMD8 for the sake of KNL, for
1713             // now..
1714 
1715             Value* vGatherResult_lo = BITCAST(EXTRACT_16(vGatherResult[0], 0), v32x8Ty);
1716             Value* vGatherResult_hi = BITCAST(EXTRACT_16(vGatherResult[0], 1), v32x8Ty);
1717 
1718             Value* vShufResult_lo = BITCAST(PSHUFB(vGatherResult_lo, vConstMask), vGatherTy);
1719             Value* vShufResult_hi = BITCAST(PSHUFB(vGatherResult_hi, vConstMask), vGatherTy);
1720 
1721             // after pshufb: group components together in each 128bit lane
1722             // 256i - 0    1    2    3    4    5    6    7
1723             //        xxxx xxxx yyyy yyyy xxxx xxxx yyyy yyyy
1724 
1725             vi128XY_lo = BITCAST(
1726                 VSHUFFLE(vShufResult_lo, vShufResult_lo, C<int32_t>({0, 1, 4, 5, 2, 3, 6, 7})),
1727                 v128bitTy);
1728             vi128XY_hi = BITCAST(
1729                 VSHUFFLE(vShufResult_hi, vShufResult_hi, C<int32_t>({0, 1, 4, 5, 2, 3, 6, 7})),
1730                 v128bitTy);
1731 
1732             // after PERMD: move and pack xy components into each 128bit lane
1733             // 256i - 0    1    2    3    4    5    6    7
1734             //        xxxx xxxx xxxx xxxx yyyy yyyy yyyy yyyy
1735         }
1736 
1737         // do the same for zw components
1738         Value* vi128ZW_lo = nullptr;
1739         Value* vi128ZW_hi = nullptr;
1740         if (isComponentEnabled(compMask, 2) || isComponentEnabled(compMask, 3))
1741         {
1742             Value* vGatherResult_lo = BITCAST(EXTRACT_16(vGatherResult[1], 0), v32x8Ty);
1743             Value* vGatherResult_hi = BITCAST(EXTRACT_16(vGatherResult[1], 1), v32x8Ty);
1744 
1745             Value* vShufResult_lo = BITCAST(PSHUFB(vGatherResult_lo, vConstMask), vGatherTy);
1746             Value* vShufResult_hi = BITCAST(PSHUFB(vGatherResult_hi, vConstMask), vGatherTy);
1747 
1748             vi128ZW_lo = BITCAST(
1749                 VSHUFFLE(vShufResult_lo, vShufResult_lo, C<int32_t>({0, 1, 4, 5, 2, 3, 6, 7})),
1750                 v128bitTy);
1751             vi128ZW_hi = BITCAST(
1752                 VSHUFFLE(vShufResult_hi, vShufResult_hi, C<int32_t>({0, 1, 4, 5, 2, 3, 6, 7})),
1753                 v128bitTy);
1754         }
1755 
1756         // init denormalize variables if needed
1757         Instruction::CastOps IntToFpCast;
1758         Value*               conversionFactor;
1759 
1760         switch (conversionType)
1761         {
1762         case CONVERT_NORMALIZED:
1763             IntToFpCast      = Instruction::CastOps::SIToFP;
1764             conversionFactor = VIMMED1((float)(1.0 / 32767.0));
1765             break;
1766         case CONVERT_SSCALED:
1767             IntToFpCast      = Instruction::CastOps::SIToFP;
1768             conversionFactor = VIMMED1((float)(1.0));
1769             break;
1770         case CONVERT_USCALED:
1771             assert(false && "Type should not be sign extended!");
1772             conversionFactor = nullptr;
1773             break;
1774         default:
1775             assert(conversionType == CONVERT_NONE);
1776             conversionFactor = nullptr;
1777             break;
1778         }
1779 
1780         // sign extend all enabled components. If we have a fill vVertexElements, output to current
1781         // simdvertex
1782         for (uint32_t i = 0; i < 4; i++)
1783         {
1784             if (isComponentEnabled(compMask, i))
1785             {
1786                 if (compCtrl[i] == ComponentControl::StoreSrc)
1787                 {
1788                     // if x or z, extract 128bits from lane 0, else for y or w, extract from lane 1
1789                     uint32_t lane = ((i == 0) || (i == 2)) ? 0 : 1;
1790                     // if x or y, use vi128XY permute result, else use vi128ZW
1791                     Value* selectedPermute_lo = (i < 2) ? vi128XY_lo : vi128ZW_lo;
1792                     Value* selectedPermute_hi = (i < 2) ? vi128XY_hi : vi128ZW_hi;
1793 
1794                     if (bFP)
1795                     {
1796                         // extract 128 bit lanes to sign extend each component
1797                         Value* temp_lo =
1798                             CVTPH2PS(BITCAST(VEXTRACT(selectedPermute_lo, C(lane)), v8x16Ty));
1799                         Value* temp_hi =
1800                             CVTPH2PS(BITCAST(VEXTRACT(selectedPermute_hi, C(lane)), v8x16Ty));
1801 
1802                         vVertexElements[currentVertexElement] = JOIN_16(temp_lo, temp_hi);
1803                     }
1804                     else
1805                     {
1806                         // extract 128 bit lanes to sign extend each component
1807                         Value* temp_lo =
1808                             PMOVSXWD(BITCAST(VEXTRACT(selectedPermute_lo, C(lane)), v8x16Ty));
1809                         Value* temp_hi =
1810                             PMOVSXWD(BITCAST(VEXTRACT(selectedPermute_hi, C(lane)), v8x16Ty));
1811 
1812                         Value* temp = JOIN_16(temp_lo, temp_hi);
1813 
1814                         // denormalize if needed
1815                         if (conversionType != CONVERT_NONE)
1816                         {
1817                             temp = FMUL(CAST(IntToFpCast, temp, mSimdFP32Ty), conversionFactor);
1818                         }
1819 
1820                         vVertexElements[currentVertexElement] = temp;
1821                     }
1822 
1823                     currentVertexElement += 1;
1824                 }
1825                 else
1826                 {
1827                     vVertexElements[currentVertexElement++] = GenerateCompCtrlVector(compCtrl[i]);
1828                 }
1829 
1830                 if (currentVertexElement > 3)
1831                 {
1832                     StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
1833                     // reset to the next vVertexElement to output
1834                     currentVertexElement = 0;
1835                 }
1836             }
1837         }
1838     }
1839     // else zero extend
1840     else if ((extendType == Instruction::CastOps::ZExt) ||
1841              (extendType == Instruction::CastOps::UIToFP))
1842     {
1843         // pshufb masks for each component
1844         Value* vConstMask[2];
1845 
1846         if (isComponentEnabled(compMask, 0) || isComponentEnabled(compMask, 2))
1847         {
1848             // x/z shuffle mask
1849             vConstMask[0] = C<char>({
1850                 0, 1, -1, -1, 4, 5, -1, -1, 8, 9, -1, -1, 12, 13, -1, -1,
1851                 0, 1, -1, -1, 4, 5, -1, -1, 8, 9, -1, -1, 12, 13, -1, -1,
1852             });
1853         }
1854 
1855         if (isComponentEnabled(compMask, 1) || isComponentEnabled(compMask, 3))
1856         {
1857             // y/w shuffle mask
1858             vConstMask[1] = C<char>({2, 3, -1, -1, 6, 7, -1, -1, 10, 11, -1, -1, 14, 15, -1, -1,
1859                                      2, 3, -1, -1, 6, 7, -1, -1, 10, 11, -1, -1, 14, 15, -1, -1});
1860         }
1861 
1862         // init denormalize variables if needed
1863         Instruction::CastOps fpCast;
1864         Value*               conversionFactor;
1865 
1866         switch (conversionType)
1867         {
1868         case CONVERT_NORMALIZED:
1869             fpCast           = Instruction::CastOps::UIToFP;
1870             conversionFactor = VIMMED1((float)(1.0 / 65535.0));
1871             break;
1872         case CONVERT_USCALED:
1873             fpCast           = Instruction::CastOps::UIToFP;
1874             conversionFactor = VIMMED1((float)(1.0f));
1875             break;
1876         case CONVERT_SSCALED:
1877             SWR_INVALID("Type should not be zero extended!");
1878             conversionFactor = nullptr;
1879             break;
1880         default:
1881             SWR_ASSERT(conversionType == CONVERT_NONE);
1882             conversionFactor = nullptr;
1883             break;
1884         }
1885 
1886         // shuffle enabled components into lower word of each 32bit lane, 0 extending to 32 bits
1887         for (uint32_t i = 0; i < 4; i++)
1888         {
1889             if (isComponentEnabled(compMask, i))
1890             {
1891                 if (compCtrl[i] == ComponentControl::StoreSrc)
1892                 {
1893                     // select correct constMask for x/z or y/w pshufb
1894                     uint32_t selectedMask = ((i == 0) || (i == 2)) ? 0 : 1;
1895                     // if x or y, use vi128XY permute result, else use vi128ZW
1896                     uint32_t selectedGather = (i < 2) ? 0 : 1;
1897 
1898                     // SIMD16 PSHUFB isnt part of AVX-512F, so split into SIMD8 for the sake of KNL,
1899                     // for now..
1900 
1901                     Value* vGatherResult_lo = EXTRACT_16(vGatherResult[selectedGather], 0);
1902                     Value* vGatherResult_hi = EXTRACT_16(vGatherResult[selectedGather], 1);
1903 
1904                     Value* temp_lo = BITCAST(
1905                         PSHUFB(BITCAST(vGatherResult_lo, v32x8Ty), vConstMask[selectedMask]),
1906                         vGatherTy);
1907                     Value* temp_hi = BITCAST(
1908                         PSHUFB(BITCAST(vGatherResult_hi, v32x8Ty), vConstMask[selectedMask]),
1909                         vGatherTy);
1910 
1911                     // after pshufb mask for x channel; z uses the same shuffle from the second
1912                     // gather 256i - 0    1    2    3    4    5    6    7
1913                     //        xx00 xx00 xx00 xx00 xx00 xx00 xx00 xx00
1914 
1915                     Value* temp = JOIN_16(temp_lo, temp_hi);
1916 
1917                     // denormalize if needed
1918                     if (conversionType != CONVERT_NONE)
1919                     {
1920                         temp = FMUL(CAST(fpCast, temp, mSimdFP32Ty), conversionFactor);
1921                     }
1922 
1923                     vVertexElements[currentVertexElement] = temp;
1924 
1925                     currentVertexElement += 1;
1926                 }
1927                 else
1928                 {
1929                     vVertexElements[currentVertexElement++] = GenerateCompCtrlVector(compCtrl[i]);
1930                 }
1931 
1932                 if (currentVertexElement > 3)
1933                 {
1934                     StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
1935                     // reset to the next vVertexElement to output
1936                     currentVertexElement = 0;
1937                 }
1938             }
1939         }
1940     }
1941     else
1942     {
1943         SWR_INVALID("Unsupported conversion type");
1944     }
1945 }
1946 
Shuffle16bpcGather(Shuffle16bpcArgs & args)1947 void FetchJit::Shuffle16bpcGather(Shuffle16bpcArgs& args)
1948 {
1949     // Unpack tuple args
1950     Value*(&vGatherResult)[2]                       = std::get<0>(args);
1951     Value*                     pVtxOut              = std::get<1>(args);
1952     const Instruction::CastOps extendType           = std::get<2>(args);
1953     const ConversionType       conversionType       = std::get<3>(args);
1954     uint32_t&                  currentVertexElement = std::get<4>(args);
1955     uint32_t&                  outputElt            = std::get<5>(args);
1956     const ComponentEnable      compMask             = std::get<6>(args);
1957     const ComponentControl(&compCtrl)[4]            = std::get<7>(args);
1958     Value*(&vVertexElements)[4]                     = std::get<8>(args);
1959 
1960     // cast types
1961     Type* vGatherTy = getVectorType(IntegerType::getInt32Ty(JM()->mContext), mVWidth);
1962     Type* v32x8Ty   = getVectorType(mInt8Ty, mVWidth * 4); // vwidth is units of 32 bits
1963 
1964     // have to do extra work for sign extending
1965     if ((extendType == Instruction::CastOps::SExt) ||
1966         (extendType == Instruction::CastOps::SIToFP) || (extendType == Instruction::CastOps::FPExt))
1967     {
1968         // is this PP float?
1969         bool bFP = (extendType == Instruction::CastOps::FPExt) ? true : false;
1970 
1971         Type* v8x16Ty   = getVectorType(mInt16Ty, 8); // 8x16bit in a 128bit lane
1972         Type* v128bitTy = getVectorType(IntegerType::getIntNTy(JM()->mContext, 128),
1973                                           mVWidth / 4); // vwidth is units of 32 bits
1974 
1975         // shuffle mask
1976         Value* vConstMask = C<char>({0, 1, 4, 5, 8, 9, 12, 13, 2, 3, 6, 7, 10, 11, 14, 15,
1977                                      0, 1, 4, 5, 8, 9, 12, 13, 2, 3, 6, 7, 10, 11, 14, 15});
1978         Value* vi128XY    = nullptr;
1979         if (isComponentEnabled(compMask, 0) || isComponentEnabled(compMask, 1))
1980         {
1981             Value* vShufResult =
1982                 BITCAST(PSHUFB(BITCAST(vGatherResult[0], v32x8Ty), vConstMask), vGatherTy);
1983             // after pshufb: group components together in each 128bit lane
1984             // 256i - 0    1    2    3    4    5    6    7
1985             //        xxxx xxxx yyyy yyyy xxxx xxxx yyyy yyyy
1986 
1987             vi128XY = BITCAST(VPERMD(vShufResult, C<int32_t>({0, 1, 4, 5, 2, 3, 6, 7})), v128bitTy);
1988             // after PERMD: move and pack xy components into each 128bit lane
1989             // 256i - 0    1    2    3    4    5    6    7
1990             //        xxxx xxxx xxxx xxxx yyyy yyyy yyyy yyyy
1991         }
1992 
1993         // do the same for zw components
1994         Value* vi128ZW = nullptr;
1995         if (isComponentEnabled(compMask, 2) || isComponentEnabled(compMask, 3))
1996         {
1997             Value* vShufResult =
1998                 BITCAST(PSHUFB(BITCAST(vGatherResult[1], v32x8Ty), vConstMask), vGatherTy);
1999             vi128ZW = BITCAST(VPERMD(vShufResult, C<int32_t>({0, 1, 4, 5, 2, 3, 6, 7})), v128bitTy);
2000         }
2001 
2002         // init denormalize variables if needed
2003         Instruction::CastOps IntToFpCast;
2004         Value*               conversionFactor;
2005 
2006         switch (conversionType)
2007         {
2008         case CONVERT_NORMALIZED:
2009             IntToFpCast      = Instruction::CastOps::SIToFP;
2010             conversionFactor = VIMMED1((float)(1.0 / 32767.0));
2011             break;
2012         case CONVERT_SSCALED:
2013             IntToFpCast      = Instruction::CastOps::SIToFP;
2014             conversionFactor = VIMMED1((float)(1.0));
2015             break;
2016         case CONVERT_USCALED:
2017             SWR_INVALID("Type should not be sign extended!");
2018             conversionFactor = nullptr;
2019             break;
2020         default:
2021             SWR_ASSERT(conversionType == CONVERT_NONE);
2022             conversionFactor = nullptr;
2023             break;
2024         }
2025 
2026         // sign extend all enabled components. If we have a fill vVertexElements, output to current
2027         // simdvertex
2028         for (uint32_t i = 0; i < 4; i++)
2029         {
2030             if (isComponentEnabled(compMask, i))
2031             {
2032                 if (compCtrl[i] == ComponentControl::StoreSrc)
2033                 {
2034                     // if x or z, extract 128bits from lane 0, else for y or w, extract from lane 1
2035                     uint32_t lane = ((i == 0) || (i == 2)) ? 0 : 1;
2036                     // if x or y, use vi128XY permute result, else use vi128ZW
2037                     Value* selectedPermute = (i < 2) ? vi128XY : vi128ZW;
2038 
2039                     if (bFP)
2040                     {
2041                         // extract 128 bit lanes to sign extend each component
2042                         vVertexElements[currentVertexElement] =
2043                             CVTPH2PS(BITCAST(VEXTRACT(selectedPermute, C(lane)), v8x16Ty));
2044                     }
2045                     else
2046                     {
2047                         // extract 128 bit lanes to sign extend each component
2048                         vVertexElements[currentVertexElement] =
2049                             PMOVSXWD(BITCAST(VEXTRACT(selectedPermute, C(lane)), v8x16Ty));
2050 
2051                         // denormalize if needed
2052                         if (conversionType != CONVERT_NONE)
2053                         {
2054                             vVertexElements[currentVertexElement] =
2055                                 FMUL(CAST(IntToFpCast,
2056                                           vVertexElements[currentVertexElement],
2057                                           mSimdFP32Ty),
2058                                      conversionFactor);
2059                         }
2060                     }
2061                     currentVertexElement++;
2062                 }
2063                 else
2064                 {
2065                     vVertexElements[currentVertexElement++] = GenerateCompCtrlVector(compCtrl[i]);
2066                 }
2067 
2068                 if (currentVertexElement > 3)
2069                 {
2070                     StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
2071                     // reset to the next vVertexElement to output
2072                     currentVertexElement = 0;
2073                 }
2074             }
2075         }
2076     }
2077     // else zero extend
2078     else if ((extendType == Instruction::CastOps::ZExt) ||
2079              (extendType == Instruction::CastOps::UIToFP))
2080     {
2081         // pshufb masks for each component
2082         Value* vConstMask[2];
2083         if (isComponentEnabled(compMask, 0) || isComponentEnabled(compMask, 2))
2084         {
2085             // x/z shuffle mask
2086             vConstMask[0] = C<char>({
2087                 0, 1, -1, -1, 4, 5, -1, -1, 8, 9, -1, -1, 12, 13, -1, -1,
2088                 0, 1, -1, -1, 4, 5, -1, -1, 8, 9, -1, -1, 12, 13, -1, -1,
2089             });
2090         }
2091 
2092         if (isComponentEnabled(compMask, 1) || isComponentEnabled(compMask, 3))
2093         {
2094             // y/w shuffle mask
2095             vConstMask[1] = C<char>({2, 3, -1, -1, 6, 7, -1, -1, 10, 11, -1, -1, 14, 15, -1, -1,
2096                                      2, 3, -1, -1, 6, 7, -1, -1, 10, 11, -1, -1, 14, 15, -1, -1});
2097         }
2098 
2099         // init denormalize variables if needed
2100         Instruction::CastOps fpCast;
2101         Value*               conversionFactor;
2102 
2103         switch (conversionType)
2104         {
2105         case CONVERT_NORMALIZED:
2106             fpCast           = Instruction::CastOps::UIToFP;
2107             conversionFactor = VIMMED1((float)(1.0 / 65535.0));
2108             break;
2109         case CONVERT_USCALED:
2110             fpCast           = Instruction::CastOps::UIToFP;
2111             conversionFactor = VIMMED1((float)(1.0f));
2112             break;
2113         case CONVERT_SSCALED:
2114             SWR_INVALID("Type should not be zero extended!");
2115             conversionFactor = nullptr;
2116             break;
2117         default:
2118             SWR_ASSERT(conversionType == CONVERT_NONE);
2119             conversionFactor = nullptr;
2120             break;
2121         }
2122 
2123         // shuffle enabled components into lower word of each 32bit lane, 0 extending to 32 bits
2124         for (uint32_t i = 0; i < 4; i++)
2125         {
2126             if (isComponentEnabled(compMask, i))
2127             {
2128                 if (compCtrl[i] == ComponentControl::StoreSrc)
2129                 {
2130                     // select correct constMask for x/z or y/w pshufb
2131                     uint32_t selectedMask = ((i == 0) || (i == 2)) ? 0 : 1;
2132                     // if x or y, use vi128XY permute result, else use vi128ZW
2133                     uint32_t selectedGather = (i < 2) ? 0 : 1;
2134 
2135                     vVertexElements[currentVertexElement] =
2136                         BITCAST(PSHUFB(BITCAST(vGatherResult[selectedGather], v32x8Ty),
2137                                        vConstMask[selectedMask]),
2138                                 vGatherTy);
2139                     // after pshufb mask for x channel; z uses the same shuffle from the second
2140                     // gather 256i - 0    1    2    3    4    5    6    7
2141                     //        xx00 xx00 xx00 xx00 xx00 xx00 xx00 xx00
2142 
2143                     // denormalize if needed
2144                     if (conversionType != CONVERT_NONE)
2145                     {
2146                         vVertexElements[currentVertexElement] =
2147                             FMUL(CAST(fpCast, vVertexElements[currentVertexElement], mSimdFP32Ty),
2148                                  conversionFactor);
2149                     }
2150                     currentVertexElement++;
2151                 }
2152                 else
2153                 {
2154                     vVertexElements[currentVertexElement++] = GenerateCompCtrlVector(compCtrl[i]);
2155                 }
2156 
2157                 if (currentVertexElement > 3)
2158                 {
2159                     StoreVertexElements(pVtxOut, outputElt++, 4, vVertexElements);
2160                     // reset to the next vVertexElement to output
2161                     currentVertexElement = 0;
2162                 }
2163             }
2164         }
2165     }
2166     else
2167     {
2168         SWR_INVALID("Unsupported conversion type");
2169     }
2170 }
2171 
2172 //////////////////////////////////////////////////////////////////////////
2173 /// @brief Output a simdvertex worth of elements to the current outputElt
2174 /// @param pVtxOut - base address of VIN output struct
2175 /// @param outputElt - simdvertex offset in VIN to write to
2176 /// @param numEltsToStore - number of simdvertex rows to write out
2177 /// @param vVertexElements - LLVM Value*[] simdvertex to write out
StoreVertexElements(Value * pVtxOut,const uint32_t outputElt,const uint32_t numEltsToStore,Value * (& vVertexElements)[4])2178 void FetchJit::StoreVertexElements(Value*         pVtxOut,
2179                                    const uint32_t outputElt,
2180                                    const uint32_t numEltsToStore,
2181                                    Value* (&vVertexElements)[4])
2182 {
2183     SWR_ASSERT(numEltsToStore <= 4, "Invalid element count.");
2184 
2185     for (uint32_t c = 0; c < numEltsToStore; ++c)
2186     {
2187         // STORE expects FP32 x vWidth type, just bitcast if needed
2188         if (!vVertexElements[c]->getType()->getScalarType()->isFloatTy())
2189         {
2190 #if FETCH_DUMP_VERTEX
2191             PRINT("vVertexElements[%d]: 0x%x\n", {C(c), vVertexElements[c]});
2192 #endif
2193             vVertexElements[c] = BITCAST(vVertexElements[c], mSimdFP32Ty);
2194         }
2195 #if FETCH_DUMP_VERTEX
2196         else
2197         {
2198             PRINT("vVertexElements[%d]: %f\n", {C(c), vVertexElements[c]});
2199         }
2200 #endif
2201         // outputElt * 4 = offsetting by the size of a simdvertex
2202         // + c offsets to a 32bit x vWidth row within the current vertex
2203         Value* dest = GEP(pVtxOut, C(outputElt * 4 + c), nullptr, "destGEP");
2204         STORE(vVertexElements[c], dest);
2205     }
2206 }
2207 
2208 //////////////////////////////////////////////////////////////////////////
2209 /// @brief Generates a constant vector of values based on the
2210 /// ComponentControl value
2211 /// @param ctrl - ComponentControl value
GenerateCompCtrlVector(const ComponentControl ctrl)2212 Value* FetchJit::GenerateCompCtrlVector(const ComponentControl ctrl)
2213 {
2214     switch (ctrl)
2215     {
2216     case NoStore:
2217         return VUNDEF_I();
2218     case Store0:
2219         return VIMMED1(0);
2220     case Store1Fp:
2221         return VIMMED1(1.0f);
2222     case Store1Int:
2223         return VIMMED1(1);
2224     case StoreVertexId:
2225     {
2226         if (mVWidth == 16)
2227         {
2228             Type*  pSimd8FPTy = getVectorType(mFP32Ty, 8);
2229             Value* pIdLo =
2230                 BITCAST(LOAD(GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_VertexID})), pSimd8FPTy);
2231             Value* pIdHi =
2232                 BITCAST(LOAD(GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_VertexID2})), pSimd8FPTy);
2233             return JOIN_16(pIdLo, pIdHi);
2234         }
2235         else
2236         {
2237             return BITCAST(LOAD(GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_VertexID})), mSimdFP32Ty);
2238         }
2239     }
2240     case StoreInstanceId:
2241     {
2242         Value* pId = BITCAST(LOAD(GEP(mpFetchInfo, {0, SWR_FETCH_CONTEXT_CurInstance})), mFP32Ty);
2243         return VBROADCAST(pId);
2244     }
2245 
2246 
2247     case StoreSrc:
2248     default:
2249         SWR_INVALID("Invalid component control");
2250         return VUNDEF_I();
2251     }
2252 }
2253 
2254 //////////////////////////////////////////////////////////////////////////
2255 /// @brief Returns the enable mask for the specified component.
2256 /// @param enableMask - enable bits
2257 /// @param component - component to check if enabled.
isComponentEnabled(ComponentEnable enableMask,uint8_t component)2258 bool isComponentEnabled(ComponentEnable enableMask, uint8_t component)
2259 {
2260     switch (component)
2261     {
2262         // X
2263     case 0:
2264         return (enableMask & ComponentEnable::X);
2265         // Y
2266     case 1:
2267         return (enableMask & ComponentEnable::Y);
2268         // Z
2269     case 2:
2270         return (enableMask & ComponentEnable::Z);
2271         // W
2272     case 3:
2273         return (enableMask & ComponentEnable::W);
2274 
2275     default:
2276         return false;
2277     }
2278 }
2279 
2280 // Don't want two threads compiling the same fetch shader simultaneously
2281 // Has problems in the JIT cache implementation
2282 // This is only a problem for fetch right now.
2283 static std::mutex gFetchCodegenMutex;
2284 
2285 //////////////////////////////////////////////////////////////////////////
2286 /// @brief JITs from fetch shader IR
2287 /// @param hJitMgr - JitManager handle
2288 /// @param func   - LLVM function IR
2289 /// @return PFN_FETCH_FUNC - pointer to fetch code
JitFetchFunc(HANDLE hJitMgr,const HANDLE hFunc)2290 PFN_FETCH_FUNC JitFetchFunc(HANDLE hJitMgr, const HANDLE hFunc)
2291 {
2292     const llvm::Function* func    = (const llvm::Function*)hFunc;
2293     JitManager*           pJitMgr = reinterpret_cast<JitManager*>(hJitMgr);
2294     PFN_FETCH_FUNC        pfnFetch;
2295 
2296     gFetchCodegenMutex.lock();
2297     pfnFetch = (PFN_FETCH_FUNC)(pJitMgr->mpExec->getFunctionAddress(func->getName().str()));
2298     // MCJIT finalizes modules the first time you JIT code from them. After finalized, you cannot
2299     // add new IR to the module
2300     pJitMgr->mIsModuleFinalized = true;
2301 
2302 #if defined(KNOB_SWRC_TRACING)
2303     char        fName[1024];
2304     const char* funcName = func->getName().data();
2305     sprintf(fName, "%s.bin", funcName);
2306     FILE* fd = fopen(fName, "wb");
2307     fwrite((void*)pfnFetch, 1, 2048, fd);
2308     fclose(fd);
2309 #endif
2310 
2311     pJitMgr->DumpAsm(const_cast<llvm::Function*>(func), "final");
2312     gFetchCodegenMutex.unlock();
2313 
2314 
2315     return pfnFetch;
2316 }
2317 
2318 //////////////////////////////////////////////////////////////////////////
2319 /// @brief JIT compiles fetch shader
2320 /// @param hJitMgr - JitManager handle
2321 /// @param state   - fetch state to build function from
JitCompileFetch(HANDLE hJitMgr,const FETCH_COMPILE_STATE & state)2322 extern "C" PFN_FETCH_FUNC JITCALL JitCompileFetch(HANDLE hJitMgr, const FETCH_COMPILE_STATE& state)
2323 {
2324     JitManager* pJitMgr = reinterpret_cast<JitManager*>(hJitMgr);
2325 
2326     pJitMgr->SetupNewModule();
2327 
2328     FetchJit theJit(pJitMgr);
2329     HANDLE   hFunc = theJit.Create(state);
2330 
2331     return JitFetchFunc(hJitMgr, hFunc);
2332 }
2333