 ef33ec0925
			
		
	
	
		ef33ec0925
		
	
	
	
	
		
			
			This PR handles implicit promotions for intrinsics when there is no exact match, such as for example clamp(int, bool, float). In this case the int and bool will be promoted to a float, and the clamp(float, float, float) form used. These promotions can be mixed with shape conversions, e.g, clamp(int, bool2, float2). Output conversions are handled either via the existing addOutputArgumentConversion function, which this PR generalizes to handle either aggregates or unaries, or by intrinsic decomposition. If there are methods or intrinsics to be decomposed, then decomposition is responsible for any output conversions, which turns out to happen automatically in all current cases. This can be revisited once inout conversions are in place. Some cases of actual ambiguity were fixed in several tests, e.g, spv.register.autoassign.* Some intrinsics with only uint versions were expanded to signed ints natively, where the underlying AST and SPIR-V supports that. E.g, countbits. This avoids extraneous conversion nodes. A new function promoteAggregate is added, and used by findFunction. This is essentially a generalization of the "promote 1st or 2nd arg" algorithm in promoteBinary. The actual selection proceeds in three steps, as described in the comments in hlslParseContext::findFunction: 1. Attempt an exact match. If found, use it. 2. If not, obtain the operator from step 1, and promote arguments. 3. Re-select the intrinsic overload from the results of step 2.
		
			
				
	
	
		
			80 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			GLSL
		
	
	
	
	
	
			
		
		
	
	
			80 lines
		
	
	
		
			1.7 KiB
		
	
	
	
		
			GLSL
		
	
	
	
	
	
| 
 | |
| struct PS_OUTPUT { float4 color : SV_Target0; };
 | |
| 
 | |
| int    i;
 | |
| uint   u;
 | |
| float  f;
 | |
| bool   b;
 | |
| 
 | |
| int2   i2;
 | |
| uint2  u2;
 | |
| float2 f2;
 | |
| bool2  b2;
 | |
| 
 | |
| Buffer    <float>  g_tTexbfs;
 | |
| Texture1D <float4> g_tTex1df4;
 | |
| uint  upos;
 | |
| float fpos;
 | |
| 
 | |
| PS_OUTPUT main()
 | |
| {
 | |
|     // Same shapes:
 | |
| 
 | |
|     float r00 = max(b,  f);
 | |
|     uint  r01 = max(b,  u);
 | |
|     int   r02 = max(b,  i);
 | |
|     float r03 = max(i,  f);
 | |
|     float r04 = max(u,  f);
 | |
| 
 | |
|     float2 r10 = max(b2,  f2);
 | |
|     uint2  r11 = max(b2,  u2);
 | |
|     int2   r12 = max(b2,  i2);
 | |
|     float2 r13 = max(i2,  f2);
 | |
|     float2 r14 = max(u2,  f2);
 | |
| 
 | |
|     float2 r20 = clamp(i2, u2, f2);  // 3 args, converts all to best type.
 | |
|     uint2  r21 = clamp(b2, u2, b2);
 | |
|     float2 r22 = clamp(b2, f2, b2);
 | |
| 
 | |
|     // Mixed shapes:
 | |
|     float2 r30 = max(b,  f2);
 | |
|     uint2  r31 = max(b,  u2);
 | |
|     int2   r32 = max(b,  i2);
 | |
|     float2 r33 = max(i,  f2);
 | |
|     float2 r34 = max(u,  f2);
 | |
| 
 | |
|     float2 r40 = clamp(i, u2, f2);  // 3 args, converts all to best type.
 | |
|     uint2  r41 = clamp(b2, u, b2);
 | |
|     float2 r42 = clamp(b2, f, b);
 | |
|     int2   r43 = clamp(i, i2, u2);
 | |
| 
 | |
|     float r50 = g_tTexbfs.Load(upos);
 | |
|     float r51 = g_tTexbfs.Load(fpos);
 | |
| 
 | |
|     int MipLevel;
 | |
| 
 | |
|     uint WidthU;
 | |
|     uint HeightU;
 | |
|     uint ElementsU;
 | |
|     uint DepthU;
 | |
|     uint NumberOfLevelsU;
 | |
|     uint NumberOfSamplesU;
 | |
| 
 | |
|     int  WidthI;
 | |
|     int  HeightI;
 | |
|     int  ElementsI;
 | |
|     int  DepthI;
 | |
|     int  NumberOfLevelsI;
 | |
|     int  NumberOfSamplesI;
 | |
| 
 | |
|     g_tTex1df4 . GetDimensions(WidthI);
 | |
|     g_tTex1df4 . GetDimensions(6, WidthI, NumberOfLevelsU);
 | |
|     g_tTex1df4 . GetDimensions(6, WidthU, NumberOfLevelsI);
 | |
|     g_tTex1df4 . GetDimensions(6, WidthI, NumberOfLevelsI);
 | |
| 
 | |
|     // max(i2, f2);
 | |
|     PS_OUTPUT ps_output;
 | |
|     ps_output.color = r00;
 | |
|     return ps_output;
 | |
| };
 |