<div dir="ltr"><div class="gmail_quote"><div dir="ltr">On Sat, Jun 10, 2017 at 3:25 PM Jose Fonseca <<a href="mailto:jfonseca@vmware.com">jfonseca@vmware.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">I don't see how to effectively tack triangle setup into the vertex<br>
shader: vertex shader applies to vertices, where as triangle setup and<br>
bining applies to primitives. Usually, each vertex gets transformed<br>
only once with llvmpipe, no matter how many triangles refer that vertex.<br>
The only way to tack triangle setup into vertex shading would be if<br>
you processed vertices a primitive at a time. Of course one could put<br>
an if-statement to skip reprocessing a vertex that already was<br>
processed, but then you have race conditions, and no benefit of inlining.<br></blockquote><div>I was mostly thinking of non-indexed vertices.</div><div><br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
And I'm afraid that tacking rasterization too is one those things that<br>
sound great on paper, quite bad in practice. And I speak from<br>
experience: in fact llvmpipe had the last step of rasterization bolted<br>
on the fragment shaders for some time. But we took it out because it<br>
was _slower_.<br>
<br>
The issue is that if you bolt on to the shader body, you either:<br>
<br>
- inline in the shader body code for the maxmimum number of planes that<br>
(which are 7, 3 sides of triangle, plus 4 sides of a scissor rect), and<br>
waste cpu cicles going through all of those tests, even when most of the<br>
time many of those tests aren't needed<br>
<br>
- or you generate if/for blocks for each place, so you only do the<br>
needed tests, but then you have branch prediction issues...<br>
<br>
Whereas if you keep rasterization _outside_ the shader you can have<br>
specialized functions to do the rasterization based on the primitive<br>
itself: (is the triangle fully inside the scissor, you need 3 planes, if<br>
the stamp is fully inside the triangle you need zero). Essentially you<br>
can "compose" by coupling two functions calls: you call a rasterization<br>
function that's especiallized for the primitive, then a shading function<br>
that's specialized for the state (but not depends on the primitive).<br>
<br>
It makes sense: rasterization needs to be specialized for the primitive,<br>
not the graphics state; where as the shader needs to be specialized for<br>
the state.<br></blockquote><div>I am planning on generating a function for each primitive type and state combination, or I can convert all primitives into triangles and just have a function for each state. The state includes stuff like if a particular clipping/scissor equation needs to be checked. I did it that way in my proof-of-concept code by using c++ templates to do the code duplication: <a href="https://github.com/programmerjake/tiled-renderer/blob/47e09f5d711803b8e899c3669fbeae3e62c9e32c/main.cpp#L366">https://github.com/programmerjake/tiled-renderer/blob/47e09f5d711803b8e899c3669fbeae3e62c9e32c/main.cpp#L366</a></div><div><br></div><div><br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
And this is just one of those non-intuitive things that's not obvious<br>
until one actually does a lot of profiling, a lot of experimentation.<br>
And trust me, lot of time was spent fine tuning this for llvmpipe (not<br>
be me -- most of rasterization was done by Keith Whitwell.) And by<br>
throwing llvmpipe out of the window and starting a new software<br>
rendering from scratch you'd be just subscribing to do it all over again.<br>
<br>
Whereas if instead of starting from scratch, you take llvmpipe, and you<br>
rewrite/replace one component at a time, you can reach exactly the same<br>
destination you want to reach, however you'll have something working<br>
every step of the way, so when you take a bad step, you can measure<br>
performance impact, and readjust. Plus if you run out of time, you have<br>
something useful -- not yet another half finished project, which quickly<br>
will rot away.<br></blockquote><div>In the case that the project is not finished this summer, I'm still planning on working on it, just at a reduced rate. If all else fails, we will at least have a up-to-date spir-v to llvm converter that handles the glsl spir-v extensions.</div><div><br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
Regarding generating the spir-v -> scalar llvm, then do whole function<br>
vectorization, I don't think it's a bad idea per se. If was I writing<br>
llvmpipe from scratch today I'd do something like that. Especially<br>
because (scalar) LLVM IR is so pervasive in the graphics ecosistem anyway.<br>
<br>
It was only after I had tgsi -> llvm ir all done that I stumbled into<br>
<a href="http://compilers.cs.uni-saarland.de/projects/wfv/" rel="noreferrer" target="_blank">http://compilers.cs.uni-saarland.de/projects/wfv/</a> .<br>
<br>
I think the important thing here is that, once you've vectorized the<br>
shader, and you converted your "texture_sample" to<br>
"texture_sample.vector8", and your "output_merger" intrinsics to<br>
"output_merger.vector8", or you log2/exp2, you then slot the fine tuned<br>
llvmpipe code for texture sampling and blending and math, as that's were<br>
your bottle necks tend to be. Because if you plan to write all texture<br>
sampling from scratch then you need a time/clone machine to complete<br>
this in a summer; and if just use LLVM's / standard C runtime's<br>
sqrt/log2/exp2/sin/cos then it would be dead slow.<br></blockquote><div>I am planning on using c++ templates to help with a lot of the texture sampler code generation -- clang can convert it to llvm ir and then I can inline it into the appropriate places. I think that all of the non-compressed image formats should be pretty easy to handle that way, as they are all pretty similar (bits packed into a long word or members of a struct). I can implement interpolation on top of the functions to load and unpack the image elements from memory. I'd estimate that, excluding the compressed texture formats, I'd need less than 10k lines and maybe a week or two to implement it all. (Glad I don't have to implement that in C.) I am planning on compiling fdlibm with clang into llvm ir, then running my vectorization algorithm on all the functions. LLVM has a spot where you can tell it that you have optimized vectorized math intrinsics, I could add them there, or implement another lowering pass to convert the intrinsics to function calls, which can then be inlined. Hopefully, that will save most of the work needed to implement vectorized math functions. Also, llvm is already pretty good at converting vectorized sqrt intrinsics to vector sqrt instructions, which x86 sse/avx and (i think) arm neon already have.</div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">
<br>
<br>
Anyway, I hope this helps. Best of luck.<br></blockquote><div>Thanks,</div><div>Jacob Lifshay</div></div></div>