Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Interoperability constructors #91

Merged
merged 7 commits into from
Mar 13, 2021

Conversation

lgritz
Copy link
Contributor

@lgritz lgritz commented Feb 8, 2021

This is a design review, feedback requested before I proceed further.

Trying out some use of enable_if to give Vec3<T> constructor and
assignment from unknown types that "look like" 3-vectors -- they have
a operator[] that returns a T, and their size is at least
3*sizeof(T).

The main idea is that if an app has rolled its own 3-vector class, it
can seamlessly construct a Vec3 from it, assign a Vec3 to it, or pass
their type as a parameter that expects a Vec3. And if the app's custom
3-vector class employs an equivalent idiom, all those operations will
work in the reverse direction.

It also works for std::vector, std::array, and even "initializer lists",
so all of this would work:

// Your app's special snowflake custom 3-vector type.
class YourCustomVec3 { ... };

// My library has an API call where I use Imath::Vec3 as a parameter
// passing convention, because I don't know about your vector.
void myfunc (const Vec3<float>& v) { ... }

// All sorts of things people may think of as "a vector"
YourCustomVec3 myvec(1, 2, 3);
std::array<float,3> arr { 1, 2, 3 };
std::vector<float> stdvec { 1, 2, 3 };

myfunc(yourvec);            // ok
myfunc(arr);                // yep
myfunc(stdvec);             // hunky-dory
myfunc({ 1.0, 2.0, 3.0 });  // yeah, even that

This is only prototyped for Vec3 for now, but if you all like this
idea, then I will complete the work to do this for the other vector
and matrix classes.

Signed-off-by: Larry Gritz [email protected]

@lgritz
Copy link
Contributor Author

lgritz commented Feb 8, 2021

@jstone-lucasfilm

@lgritz lgritz marked this pull request as draft February 8, 2021 20:26
@lgritz
Copy link
Contributor Author

lgritz commented Feb 8, 2021

I'm a little unsure about my having marked these as constexpr14 and noexcept -- if people feel that it's is an unnecessary constraint to require the foreign vector's operator[] be constexpr and not throw exceptions, then we can relax this.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 8, 2021

You may see from my #if 0 that I couldn't quite make it work with a straight C "float[3]" style array. But there's probably some messing around with declarations that would make that work, if that's also important.

Copy link
Contributor

@meshula meshula left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is an exciting development. It would be keen if there was shenanigan-magic for C as well!

// Test construction/assignment/paramater pass of a C array
{
float s[3] = { 1, 2, 3 };
Vec3<float> v(s);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Much as I would love something like this for interoperation with my C code, I kind of wonder if it's an unsafe dead end, because the float[3] might not be reasonably distinguishable from a float*. I'd be happy enough if I got magic bridging for typedef struct { float x, y, z; } V3f;

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Unfortunately a subscript operator is not going to play out happily for this common C idiom for lightweight n-vecs

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think I can make it work with a different (second) construct that looks to see if there's a .x, .y, .z of the right types.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

After my next meeting is over, I'll give it a stab.

@meshula
Copy link
Contributor

meshula commented Feb 9, 2021

BTW, I think your proposal will work with Eigen, certainly that would be a good robustness check versus possibly the most complicated V3f declaration in the Known Universe.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 9, 2021

I'm still tinkering, trying to get it to correctly recognize either things with .x, .y, .z, or with operator[] -- without ending up with an error for a redundant declaration for a class that has both. I'm sure this can be cracked, but it's definitely keeping me on the uncomfortable bleeding edge of my understanding of template metaprogramming.

@jstone-lucasfilm
Copy link
Member

This is great to see, Larry, and I'm looking forward to integrating this approach into MaterialX as well.

Responding to your thoughts above, I would agree that it would be ideal to minimize the restrictions on the foreign vector's [] operator, which may not contain noexcept or constexpr markings. As Nick mentions, testing against Eigen would be a strong proof-of-concept, since its vector/matrix classes are so fundamentally different from Imath.

For helpers such as IMATH_VECTOR_EQUIVALENT, it might be more idiomatic to use an alias template (e.g. template <class T> using imath_vector_equivalent), but the current macro should work fine if it's aligned with the approach elsewhere in the Imath library.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 9, 2021

Thanks for the tips. Yes, I'm working on those templates now. Got frustrated with template error messages last night, hopefully fresh eyes today will let me figure out how to make it happy.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 12, 2021

Got it! Including using C arrays, classes that have both named members and subscripts, and a way to override false positives and negatives.

Will update as soon as I'm done with dentist appointment.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 12, 2021

Updated with a revised approach:

  • New file ImathTypeTraits.h now collects all the type traits (including
    a couple things we had previously put in ImathPlatform.h).

  • New traits: has_subscript, has_xy, has_xyz, has_xyzw.

  • New test file: testInterop.{h,cpp}

  • Plain C arrays[3] work now.

  • Classes that have .x, .y, .z member variables (but no subscripting)
    work now.

  • Classes that have BOTH subscripting and named members work!

  • It's easy to correct false positives and false negatives.

I still have only prototyped for Vec3, if people like the design, I will
extend to the other relevant types.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 12, 2021

If any of you have custom vector types lying around in your garage, please try this patch out and verify that it works to allow you to assign your vector type to a V3f, and to pass your vector type to a function that takes a const V3f& as a parameter.

@meshula
Copy link
Contributor

meshula commented Feb 12, 2021

Love it! Here's another goofy thing to add to the tests as a case which should trait out as NO. I'm curious how the tests hold up when functions exist named as the components.

template<typename T>
struct AccessedV3 {
    T x() const;
   T y() const;
   T z() const;    
};

@lgritz
Copy link
Contributor Author

lgritz commented Feb 12, 2021

Love it! Here's another goofy thing to add to the tests as a case which should trait out as NO. I'm curious how the tests hold up when functions exist named as the components.

Already was there -- look in testInterop.cpp for the xyz_wrong class. I suppose we could make that work, too, if people thought it was important?

@lgritz
Copy link
Contributor Author

lgritz commented Feb 12, 2021

Oops, I seem to be failing CI for some compilers, will need some minor adjustments, will fix.

@lgritz lgritz force-pushed the lg-genericvec branch 6 times, most recently from 56d931d to 29d33dc Compare February 13, 2021 00:26
@meshula
Copy link
Contributor

meshula commented Feb 13, 2021

Oh! I missed that when I read the diff. Eigen supports subscripting and the x() syntax, so I think subscripting covers it. I don't know of other vector libs that do that particularly, so I don't know of a reason to capture it as a Yes case.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 13, 2021

@meshula If you're a regular Eigen user, can you give this a quick try and see if it works? I could fumble around with it, but still not be sure I was using the idioms you care about, and I'm sure what would take me hours to set up is something you can do in minutes if Eigen is in your daily toolbox.

@cary-ilm
Copy link
Member

Should we merge this as is and continue to iterate on it with subsequent PR's? Might be easier to test that way, and it's not likely to break anything.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 13, 2021

I consider this incomplete, and I'd really like to extend this work to the other classes before we merge anything.

Does everybody know about the GitHub command line app?

gh pr checkout 91

will correctly figure out this PR (#91), pull from my tree into your local copy, and let you try it out. No need to even set up remotes by hand or know where my repo is. There is no need to merge into our main repo as a prerequisite for people trying it out super easily.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 13, 2021

I'll flesh this out to the other classes over the weekend and turn into a full PR. For this draft, I was seeking to give enough of a peek that people could give feedback (pro or con) on the basic idea, do we want this, does the design seem sane, etc.

I think today's iteration is looking pretty good (IMHO) but this is a new idiom for us and I don't want to force people into it if they think it unwise or have reservations about my approach to implementation.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 27, 2021

I don't quite understand how the new constructors are related to the string initialization.
This compiles cleanly on the current Imath master, but not against the changes of my patch?
Do you have any insight, or are you presuming this is a compiler bug?

I'm fine with disabling for gcc4, which in my world anyway is about to become irrelevant within weeks (at the very least, anything still depending on it is locked down on legacy versions of the whole stack and also would not be upgrading to Imath3).

If we continue to have lingering suspicions of these new constructors, I'm very happy to do something like this:

#ifndef IMATH_FOREIGN_VECTOR_INTEROP
#    define IMATH_FOREIGN_VECTOR_INTEROP 1
#endif

#if IMATH_FOREIGN_VECTOR_INTEROP
    ... my new ctrs here
#endif

and that give an escape valve for somebody to

#define IMATH_FOREIGN_VECTOR_INTEROP 0
#include <Imath/ImathVec.h>

if they are building software or using a compiler that has trouble and just wants to disable the interop constructors.

I'm also thinking of declaring them as explicit, which means that people are less likely to use them unintentionally, but at the expense of getting a compile error if they try to pass or assign implicitly, i.e. myfunc_taking_imath(othervec) would need to be myfunc_taking_imath(Imath::V3f(othervec)), slightly wordier but fewer changes for semantic ambiguity.

@cary-ilm
Copy link
Member

I encountered the problem while building Field3D, then reconstructed the offending construct inside testInterop.cpp, where simply adding the map<string,v3f> leads to the error. It looks like it's getting confused inside the has_xyz<> template. The complete error is below. This is gcc 4.8.5 with -std=c++1y. Since the same code works with gcc6, it does look like a compiler bug.

Providing an option to opt out seems like a good idea. These constructs are fancy enough that they may well trip up other compilers we're not explicitly validating.

In file included from /usr/include/c++/4.8.2/bits/stl_map.h:63:0,
from /usr/include/c++/4.8.2/map:61,
from lib/base/ImathTest/testInterop.cpp:18:
/usr/include/c++/4.8.2/tuple: In instantiation of ‘constexpr std::_Head_base<_Idx, _Head, false>::_Head_base() [with long unsigned int _Idx = 0ul; _Head = const std::basic_string&]’:
/usr/include/c++/4.8.2/tuple:253:29: required from ‘constexpr std::_Tuple_impl<_Idx, _Head, _Tail ...>::_Tuple_impl() [with long unsigned int _Idx = 0ul; _Head = const std::basic_string&; _Tail = {}]’
/usr/include/c++/4.8.2/tuple:396:20: required from ‘constexpr std::tuple< >::tuple() [with _Elements = {const std::basic_string<char, std::char_traits, std::allocator >&}]’
lib/base/Imath/ImathTypeTraits.h:113:14: required from ‘struct Imath::has_xyz<std::tuple<const std::basic_string<char, std::char_traits, std::allocator >&>, float>’
lib/base/Imath/ImathVec.h:324:26: required from ‘std::_Rb_tree_node<_Val>::_Rb_tree_node(_Args&& ...) [with _Args = {const std::piecewise_construct_t&, std::tuple<const std::basic_string<char, std::char_traits, std::allocator >&>, std::tuple<>}; _Val = std::pair<const std::basic_string, Imath::Vec3 >]’
/usr/include/c++/4.8.2/ext/new_allocator.h:120:4: required from ‘void __gnu_cxx::new_allocator<_Tp>::construct(_Up*, _Args&& ...) [with _Up = std::_Rb_tree_node<std::pair<const std::basic_string, Imath::Vec3 > >; _Args = {const std::piecewise_construct_t&, std::tuple<const std::basic_string<char, std::char_traits, std::allocator >&>, std::tuple<>}; _Tp = std::_Rb_tree_node<std::pair<const std::basic_string, Imath::Vec3 > >]’
/usr/include/c++/4.8.2/bits/alloc_traits.h:254:4: required from ‘static typename std::enable_if<std::allocator_traits<_Alloc>::__construct_helper<_Tp, _Args>::value, void>::type std::allocator_traits<_Alloc>::_S_construct(_Alloc&, _Tp*, _Args&& ...) [with _Tp = std::_Rb_tree_node<std::pair<const std::basic_string, Imath::Vec3 > >; _Args = {const std::piecewise_construct_t&, std::tuple<const std::basic_string<char, std::char_traits, std::allocator >&>, std::tuple<>}; _Alloc = std::allocator<std::_Rb_tree_node<std::pair<const std::basic_string, Imath::Vec3 > > >; typename std::enable_if<std::allocator_traits<_Alloc>::__construct_helper<_Tp, _Args>::value, void>::type = void]’
/usr/include/c++/4.8.2/bits/alloc_traits.h:393:57: required from ‘static decltype (_S_construct(__a, __p, (forward<_Args>)(std::allocator_traits::construct::__args)...)) std::allocator_traits<_Alloc>::construct(_Alloc&, _Tp*, _Args&& ...) [with _Tp = std::_Rb_tree_node<std::pair<const std::basic_string, Imath::Vec3 > >; _Args = {const std::piecewise_construct_t&, std::tuple<const std::basic_string<char, std::char_traits, std::allocator >&>, std::tuple<>}; _Alloc = std::allocator<std::_Rb_tree_node<std::pair<const std::basic_string, Imath::Vec3 > > >; decltype (_S_construct(__a, __p, (forward<_Args>)(std::allocator_traits::construct::__args)...)) = ]’
/usr/include/c++/4.8.2/bits/stl_tree.h:408:36: required from ‘std::_Rb_tree_node<_Val>* std::_Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::_M_create_node(_Args&& ...) [with _Args = {const std::piecewise_construct_t&, std::tuple<const std::basic_string<char, std::char_traits, std::allocator >&>, std::tuple<>}; _Key = std::basic_string; _Val = std::pair<const std::basic_string, Imath::Vec3 >; _KeyOfValue = std::_Select1st<std::pair<const std::basic_string, Imath::Vec3 > >; _Compare = std::less<std::basic_string >; _Alloc = std::allocator<std::pair<const std::basic_string, Imath::Vec3 > >; std::_Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::_Link_type = std::_Rb_tree_node<std::pair<const std::basic_string, Imath::Vec3 > >*]’
/usr/include/c++/4.8.2/bits/stl_tree.h:1669:64: required from ‘std::_Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::iterator std::_Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::_M_emplace_hint_unique(std::_Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::const_iterator, _Args&& ...) [with _Args = {const std::piecewise_construct_t&, std::tuple<const std::basic_string<char, std::char_traits, std::allocator >&>, std::tuple<>}; _Key = std::basic_string; _Val = std::pair<const std::basic_string, Imath::Vec3 >; _KeyOfValue = std::_Select1st<std::pair<const std::basic_string, Imath::Vec3 > >; _Compare = std::less<std::basic_string >; _Alloc = std::allocator<std::pair<const std::basic_string, Imath::Vec3 > >; std::_Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::iterator = std::_Rb_tree_iterator<std::pair<const std::basic_string, Imath::Vec3 > >; std::_Rb_tree<_Key, _Val, _KeyOfValue, _Compare, _Alloc>::const_iterator = std::_Rb_tree_const_iterator<std::pair<const std::basic_string, Imath::Vec3 > >]’
/usr/include/c++/4.8.2/bits/stl_map.h:465:8: required from ‘std::map<_Key, _Tp, _Compare, _Alloc>::mapped_type& std::map<_Key, _Tp, _Compare, _Alloc>::operator[](const key_type&) [with _Key = std::basic_string; _Tp = Imath::Vec3; _Compare = std::less<std::basic_string >; _Alloc = std::allocator<std::pair<const std::basic_string, Imath::Vec3 > >; std::map<_Key, _Tp, _Compare, _Alloc>::mapped_type = Imath::Vec3; std::map<_Key, _Tp, _Compare, _Alloc>::key_type = std::basic_string]’
lib/base/ImathTest/testInterop.cpp:718:11: required from here
/usr/include/c++/4.8.2/tuple:132:22: error: value-initialization of reference type ‘const std::basic_string&’
: _M_head_impl() { }

@lgritz
Copy link
Contributor Author

lgritz commented Feb 27, 2021

OK, two questions, then:

  1. Add guards to opt out, or opt in?

  2. Should the disabling be for gcc < 6, or for C++ < 14? (Both will include users that are anywhere near the current VFX Platform stack, exclude something old).

@cary-ilm
Copy link
Member

  1. Let's make it in by default, with the ability to turn it off if there's an error. This runs the risk of confusing people who get an obscure error message, but we can make note of it in the installation guide. But at least that promotes the functionality.

2.The failure I'm seeing depends on GCC4, which claims to support c++11 but apparently doesn't. Since the code conforms to C++11 as written, it seems reasonable to disable based on GCC4. I'm also seeing similar sorts of failures in code that has to be compiled with CUDA 6.2.

@lgritz
Copy link
Contributor Author

lgritz commented Feb 28, 2021

Updated the PR:

  • Disable interop if the symbol IMATH_FOREIGN_VECTOR_INTEROP is defined to be 0 prior to including any Imath headers.

  • Also disable if we detect gcc 4.x, which seems to have a bug that prevents this idiom from compiling correctly.

  • Corral the interop functionality into a documentation "group" and beef up the description.

@lgritz lgritz changed the title RFC: Interoperability constructors Interoperability constructors Feb 28, 2021
@lgritz
Copy link
Contributor Author

lgritz commented Feb 28, 2021

I would request that you try again with these modifications and see if your code base is safe.

Assuming that is true, I stand by this version of the PR and you can merge when it passes your test.

@kdt3rd
Copy link
Contributor

kdt3rd commented Mar 1, 2021

So this is cool - type traits are cool.

However, I question this change a bit. I worry about unintended object slicing because we are implicitly doing this - basically allowing people to be too loose with their types. I get that this is the point, but I worry about putting a Vec4 into a Vec3 and people not projecting that in explicitly to say "yes, I know, it's fine".

Second, the sizeof check in the has_subscript stuff seems ... flawed. It is bypassing too much of the type system to my mind to be a viable thing (the has_xyz stuff is good stuff, given the above caveat). For example, if I do:

struct MyPainInTheNeck
{
float &operator[](int i) { return *( ((float *)this) + i); }

float x;
float y;
int32_t flags;
};

this will pass the "has_subscript and ! has_xyz" check for Vec3, right? And not even crash. But I contend that the z value would not be what one might expect :)

So, I like the idea of easier type coersion, but it seems like we are running too fast and loose and we run the risk of causing more pain for people than we should. Should we change the has_subscript stuff to not do sizeof checks for the default implementation, but instead check if the type has a member function called size, returning an integral type, and instead only do the sizeof checks in the template specialization for the Base[N] implementation (nothing says they need to have the same implementation)? Then you would probably have to make an adapter function to access values, where the one checks the size first... But this then should work for vector, et al. without further specialization, right?

Another option this to make these constructors marked as explicit, such that someone has to explictly say "yes, I want you to do the conversion". It'd at least still be foo = Vec3( othertype ); instead.

Finally, might want to add support for tuples, and use std::tie to extract values, at least for the vec types, such that you can do std::tie( x, y, z ) = foo, where foo is of type std::tuple<T, T, T> for a vec3, or similar. I don't know if that helps much for the above, I think if anything it would just move the problem, but might be a handy variant to support.

@lgritz
Copy link
Contributor Author

lgritz commented Mar 1, 2021

@kdt3rd -- yeah, I think explicit is the safer way to go. I will fix that today.

@lgritz
Copy link
Contributor Author

lgritz commented Mar 1, 2021

Pushed update -- explicit interop constructors only.

lgritz added a commit to AcademySoftwareFoundation/OpenImageIO that referenced this pull request Mar 1, 2021
I have submitted a PR to Imath that adds "interopability" constructors
and assignment, that use some templates to allow seamless
construction, assignment, and passing of "foreign" vector types to
Imath vectors. This makes it easier library A that uses Imath vector
types in its API to interact with app or library B that has a custom
internal vector type, without a ton of ugly casts and copies.
(See AcademySoftwareFoundation/Imath#91)

These created a few minor conflicts with OIIO's internal SIMD classes
and their construction and casting from Imath types. This patch cleans
up the ambiguities:

* Make Imath::V4f -> simd::vfloat4 and Imath::M44f -> simd::matrix44
  constructor `explicit`.

* Make the simd types `simd()` method (returning the underlying raw SIMD
  type) also have a variety that returns a ref to a non-const vector,
  to ensure proper working of the _MM_TRANSPOSE4_PS macro.

Now I am going to commit a sin. This MUST get backported to OIIO 2.2
in order for it to not break against the upcoming Imath 3.0.  This
doesn't break ABI compatibility, nor change OIIO's primary advertised
public APIs, but it does change the technically/informally public APIs
of simd.h in ways that introduce the possibility that an app using
those simd vector classes *might* need minor source code editing if
they relied on those implicit constructors. Such software might need
certain `simd::vector4 foo = imath_bar` to become `foo = vector4(bar)`
and the same for certain matrix44 assignments.  It is usually a big
no-no to backport any change that could require editing of source
code.  But since the alternative is not being able to use the new
Imath, I think this is the lesser evil, and maybe nobody out there is
doing the thing that would need to be changed?
lgritz added a commit to lgritz/OpenImageIO that referenced this pull request Mar 1, 2021
…emySoftwareFoundation#2878)

I have submitted a PR to Imath that adds "interopability" constructors
and assignment, that use some templates to allow seamless
construction, assignment, and passing of "foreign" vector types to
Imath vectors. This makes it easier library A that uses Imath vector
types in its API to interact with app or library B that has a custom
internal vector type, without a ton of ugly casts and copies.
(See AcademySoftwareFoundation/Imath#91)

These created a few minor conflicts with OIIO's internal SIMD classes
and their construction and casting from Imath types. This patch cleans
up the ambiguities:

* Make Imath::V4f -> simd::vfloat4 and Imath::M44f -> simd::matrix44
  constructor `explicit`.

* Make the simd types `simd()` method (returning the underlying raw SIMD
  type) also have a variety that returns a ref to a non-const vector,
  to ensure proper working of the _MM_TRANSPOSE4_PS macro.

Now I am going to commit a sin. This MUST get backported to OIIO 2.2
in order for it to not break against the upcoming Imath 3.0.  This
doesn't break ABI compatibility, nor change OIIO's primary advertised
public APIs, but it does change the technically/informally public APIs
of simd.h in ways that introduce the possibility that an app using
those simd vector classes *might* need minor source code editing if
they relied on those implicit constructors. Such software might need
certain `simd::vector4 foo = imath_bar` to become `foo = vector4(bar)`
and the same for certain matrix44 assignments.  It is usually a big
no-no to backport any change that could require editing of source
code.  But since the alternative is not being able to use the new
Imath, I think this is the lesser evil, and maybe nobody out there is
doing the thing that would need to be changed?
src/Imath/ImathTypeTraits.h Outdated Show resolved Hide resolved
src/Imath/ImathTypeTraits.h Outdated Show resolved Hide resolved
@meshula
Copy link
Contributor

meshula commented Mar 2, 2021

With regards MaterialX, construction works, but assignment doesn't. There may be other issues, but this seems like a good one to look at before going deeper. MatX vector hides the data in a std::array<T, N>.

Here's an example of the output for assignment and a trivial repro

 void testImath_matx()
{
     MaterialX::Vector4 mxv4 = {1,2,3,4};
     Imath::V4f imv4 = mxv4;
}

[ 84%] Building CXX object source/MaterialXView/CMakeFiles/MaterialXView.dir/Viewer.cpp.o
/Users/dp/Projects/MaterialX/source/MaterialXView/Viewer.cpp:62:16: error: no
      viable conversion from 'MaterialX::Vector4' to 'Imath::V4f'
      (aka 'Vec4<float>')
    Imath::V4f imv4 = mxv4;
               ^      ~~~~
/Users/dp/Projects/MaterialX/source/MaterialXView/../ImathVec.h:602:40: note:
      candidate constructor not viable: no known conversion from
      'MaterialX::Vector4' to 'const Imath_3_0::Vec4<float> &' for 1st argument
    IMATH_HOSTDEVICE IMATH_CONSTEXPR14 Vec4 (const Vec4& v) noexcept;
                                       ^
/Users/dp/Projects/MaterialX/source/MaterialXView/../ImathVec.h:605:59: note:
      candidate template ignored: could not match 'Vec4<type-parameter-0-0>'
      against 'MaterialX::Vector4'
    template <class S> IMATH_HOSTDEVICE IMATH_CONSTEXPR14 Vec4 (const Ve...
                                                          ^
1 error generated.

@lgritz
Copy link
Contributor Author

lgritz commented Mar 2, 2021

Can you try with the latest, in which I've made the constructors explicit?

lgritz added 7 commits March 12, 2021 19:10
This is a design review, feedback requested before I proceed further.

Trying out some use of enable_if to give `Vec3<T>` constructor and
assignment from unknown types that "look like" 3-vectors -- they have
a `operator[]` that returns a T, and their size is at least
3*sizeof(T).

The main idea is that if an app has rolled its own 3-vector class, it
can seamlessly construct a Vec3 from it, assign a Vec3 to it, or pass
their type as a parameter that expects a Vec3. And if the app's custom
3-vector class employs an equivalent idiom, all those operations will
work in the reverse direction.

It also works for std::vector, std::array, and even "initializer lists",
so all of this would work:

    // Your app's special snowflake custom 3-vector type.
    class YourCustomVec3 { ... };

    // My library has an API call where I use Imath::Vec3 as a parameter
    // passing convention, because I don't know about your vector.
    void myfunc (const Vec3<float>& v) { ... }

    // All sorts of things people may think of as "a vector"
    YourCustomVec3 myvec(1, 2, 3);
    std::array<float,3> arr { 1, 2, 3 };
    std::vector<float> stdvec { 1, 2, 3 };

    myfunc(yourvec);            // ok
    myfunc(arr);                // yep
    myfunc(stdvec);             // hunky-dory
    myfunc({ 1.0, 2.0, 3.0 });  // yeah, even that

This is only prototyped for Vec3 for now, but if you all like this
idea, then I will complete the work to do this for the other vector
and matrix classes.

Signed-off-by: Larry Gritz <[email protected]>
Different approach:

* New file ImathTypeTraits.h now collects all the type traits (including
  a couple things we had previously put in ImathPlatform.h).

* New traits: has_subscript, has_xy, has_xyz, has_xyzw.

* New test file: testInterop.{h,cpp}

* Plain C arrays[3] work now.

* Classes that have .x, .y, .z member variables (but no subscripting)
  work now.

* Classes that have BOTH subscripting and named members work!

* It's easy to correct false positives and false negatives.

I still have only prototyped for Vec3, if people like the design, I will
extend to the other relevant types.

Signed-off-by: Larry Gritz <[email protected]>
Also change the ones that require subscripting to non-constexpr,
non-noexcept, to accommodate interop with foreign types whose
operator[] is not constexpr or noexcept.  I have mixed feelings about
this; most sane implementations *ought* to be constexpr and noexcept,
but I can imagine that many are not so declared even though they could
be. I have left the .xyzw direct accessors constexpr and noexcept,
because simple assignment of these member variables ought to be safe.

Removed unused isSameType templates (if needed, we should use
`std::is_same<>`).

Signed-off-by: Larry Gritz <[email protected]>
* Disable interop if the symbol IMATH_FOREIGN_VECTOR_INTEROP is
  defined to be 0 prior to including any Imath headers.

* Also disable if we detect gcc 4.x, which seems to have a bug that
  prevents this idiom from compiling correctly.

* Corral the interop functionality into a documentation "group" and
  beef up the description.

Signed-off-by: Larry Gritz <[email protected]>
Signed-off-by: Larry Gritz <[email protected]>
@lgritz
Copy link
Contributor Author

lgritz commented Mar 13, 2021

I just rebased this to the current master. We decided in TSC to go ahead and merge so we include this in 3.0, right?

@cary-ilm
Copy link
Member

cary-ilm commented Mar 13, 2021 via email

@lgritz
Copy link
Contributor Author

lgritz commented Mar 13, 2021

OK, will do as soon as the CI is done.

@lgritz lgritz merged commit 967ec17 into AcademySoftwareFoundation:master Mar 13, 2021
@lgritz lgritz deleted the lg-genericvec branch March 13, 2021 05:51
@cary-ilm cary-ilm added the build label Mar 18, 2021
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants