Skip to content

Commit

Permalink
ShadingEngine : Support 64 bit ints in `RendererServices::get_userdat…
Browse files Browse the repository at this point in the history
…a()`

This requires ImageEngine/cortex#1439 to work.
  • Loading branch information
johnhaddon committed Nov 4, 2024
1 parent a9f0121 commit d63d0ea
Show file tree
Hide file tree
Showing 3 changed files with 60 additions and 18 deletions.
1 change: 1 addition & 0 deletions Changes.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ Fixes

- Render, InteractiveRender : Added default node name arguments to the compatibility shims for removed subclasses such as ArnoldRender.
- GafferUITest : Fixed `assertNodeUIsHaveExpectedLifetime()` test for invisible nodes.
- OSLObject : Fixed `getattribute()` to support 64 bit integer data, such as an `instanceId` primitive variable loaded from USD. Since OSL doesn't provide a 64 bit integer type, values are truncated to 32 bits.

1.5.0.1 (relative to 1.5.0.0)
=======
Expand Down
27 changes: 27 additions & 0 deletions python/GafferOSLTest/ShadingEngineTest.py
Original file line number Diff line number Diff line change
Expand Up @@ -160,6 +160,33 @@ def testDoubleAsIntViaGetAttribute( self ) :
for i, c in enumerate( p["Ci"] ) :
self.assertEqual( c[0], float(i) )

def testInt64GetAttribute( self ) :

shader = self.compileShader( pathlib.Path( __file__ ).parent / "shaders" / "intAttribute.osl" )

for dataType in ( IECore.Int64VectorData, IECore.UInt64VectorData ) :

with self.subTest( dataType = dataType ) :

points = IECore.CompoundData( {
"P" : IECore.V3fVectorData( [ imath.V3f( i ) for i in range( 0, 10 ) ] ),
"int64Data" : dataType( range( 0, 10 ) )
} )

engine = GafferOSL.ShadingEngine( IECoreScene.ShaderNetwork(
shaders = {
"output" : IECoreScene.Shader( shader, "osl:surface", { "name" : "int64Data" } ),
},
output = "output"
) )

self.assertTrue( engine.needsAttribute( "int64Data" ) )

points = engine.shade( points )

for i, c in enumerate( points["Ci"] ) :
self.assertEqual( c[0], float( i ) )

def testUserDataViaGetAttribute( self ) :

shader = self.compileShader( pathlib.Path( __file__ ).parent / "shaders" / "attribute.osl" )
Expand Down
50 changes: 32 additions & 18 deletions src/GafferOSL/ShadingEngine.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -215,6 +215,29 @@ DataPtr dataFromTypeDesc( TypeDesc type, void *&basePointer )
return nullptr;
}

template<typename SourceType>
bool convertScalar( void *dst, TypeDesc dstType, const void *src )
{
const SourceType *typedSrc = reinterpret_cast<const SourceType *>( src );
if( dstType == TypeDesc::FLOAT )
{
if( typedSrc && dst )
{
*((float*)dst) = static_cast<float>( *typedSrc );
}
return true;
}
else if( dstType == TypeDesc::INT )
{
if( typedSrc && dst )
{
*((int*)dst) = static_cast<int>( *typedSrc );
}
return true;
}
return false;
}

// Equivalent to `OSL::ShadingSystem:convert_value()`, but with support for
// additional conversions.
bool convertValue( void *dst, TypeDesc dstType, const void *src, TypeDesc srcType )
Expand Down Expand Up @@ -254,24 +277,15 @@ bool convertValue( void *dst, TypeDesc dstType, const void *src, TypeDesc srcTyp
}
else if( srcType.basetype == TypeDesc::DOUBLE && srcType.aggregate == TypeDesc::SCALAR )
{
const double *doubleCast = reinterpret_cast<const double *>( src );
if( dstType == TypeDesc::FLOAT )
{
if( doubleCast && dst )
{
*((float*)dst) = static_cast<float>( *doubleCast );
}
return true;
}
else if( dstType == TypeDesc::INT )
{
if( doubleCast && dst )
{
*((int*)dst) = static_cast<int>( *doubleCast );
}
return true;
}
return false;
return convertScalar<double>( dst, dstType, src );
}
else if( srcType.basetype == TypeDesc::INT64 && srcType.aggregate == TypeDesc::SCALAR )
{
return convertScalar<int64_t>( dst, dstType, src );
}
else if( srcType.basetype == TypeDesc::UINT64 && srcType.aggregate == TypeDesc::SCALAR )
{
return convertScalar<uint64_t>( dst, dstType, src );
}

return false;
Expand Down

0 comments on commit d63d0ea

Please sign in to comment.