Home | History | Annotate | Download | only in Eigen
      1 // This file is part of Eigen, a lightweight C++ template library
      2 // for linear algebra.
      3 //
      4 // Copyright (C) 2010 Gael Guennebaud <gael.guennebaud (a] inria.fr>
      5 //
      6 // This Source Code Form is subject to the terms of the Mozilla
      7 // Public License v. 2.0. If a copy of the MPL was not distributed
      8 // with this file, You can obtain one at http://mozilla.org/MPL/2.0/.
      9 
     10 #ifndef EIGEN_OPENGL_MODULE
     11 #define EIGEN_OPENGL_MODULE
     12 
     13 #include <Eigen/Geometry>
     14 
     15 #if defined(__APPLE_CC__)
     16   #include <OpenGL/gl.h>
     17 #else
     18   #include <GL/gl.h>
     19 #endif
     20 
     21 namespace Eigen {
     22 
     23 /**
     24   * \defgroup OpenGLSUpport_Module OpenGL Support module
     25   *
     26   * This module provides wrapper functions for a couple of OpenGL functions
     27   * which simplify the way to pass Eigen's object to openGL.
     28   * Here is an exmaple:
     29   * 
     30   * \code
     31   * // You need to add path_to_eigen/unsupported to your include path.
     32   * #include <Eigen/OpenGLSupport>
     33   * // ...
     34   * Vector3f x, y;
     35   * Matrix3f rot;
     36   * 
     37   * glVertex(y + x * rot);
     38   * 
     39   * Quaternion q;
     40   * glRotate(q);
     41   * 
     42   * // ...
     43   * \endcode
     44   *
     45   */
     46 //@{
     47 
     48 #define EIGEN_GL_FUNC_DECLARATION(FUNC)                                                                             \
     49 namespace internal {                                                                                                \
     50   template< typename XprType,                                                                                       \
     51             typename Scalar = typename XprType::Scalar,                                                             \
     52             int Rows = XprType::RowsAtCompileTime,                                                                  \
     53             int Cols = XprType::ColsAtCompileTime,                                                                  \
     54             bool IsGLCompatible = bool(XprType::Flags&LinearAccessBit)                                              \
     55                               && bool(XprType::Flags&DirectAccessBit)                                               \
     56                               && (XprType::IsVectorAtCompileTime || (XprType::Flags&RowMajorBit)==0)>               \
     57   struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl);                                                                      \
     58                                                                                                                     \
     59   template<typename XprType, typename Scalar, int Rows, int Cols>                                                   \
     60   struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<XprType,Scalar,Rows,Cols,false> {                                     \
     61     inline static void run(const XprType& p) {                                                                      \
     62       EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<typename plain_matrix_type_column_major<XprType>::type>::run(p); }       \
     63   };                                                                                                                \
     64 }                                                                                                                   \
     65                                                                                                                     \
     66 template<typename Derived> inline void FUNC(const Eigen::DenseBase<Derived>& p) {                                   \
     67   EIGEN_CAT(EIGEN_CAT(internal::gl_,FUNC),_impl)<Derived>::run(p.derived());                                        \
     68 }
     69 
     70 
     71 #define EIGEN_GL_FUNC_SPECIALIZATION_MAT(FUNC,SCALAR,ROWS,COLS,SUFFIX)                                              \
     72 namespace internal {                                                                                                \
     73   template< typename XprType> struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<XprType, SCALAR, ROWS, COLS, true> {      \
     74     inline static void run(const XprType& p) { FUNC##SUFFIX(p.data()); }                                            \
     75   };                                                                                                                \
     76 }
     77 
     78   
     79 #define EIGEN_GL_FUNC_SPECIALIZATION_VEC(FUNC,SCALAR,SIZE,SUFFIX)                                                   \
     80 namespace internal {                                                                                                \
     81   template< typename XprType> struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<XprType, SCALAR, SIZE, 1, true> {         \
     82     inline static void run(const XprType& p) { FUNC##SUFFIX(p.data()); }                                            \
     83   };                                                                                                                \
     84   template< typename XprType> struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<XprType, SCALAR, 1, SIZE, true> {         \
     85     inline static void run(const XprType& p) { FUNC##SUFFIX(p.data()); }                                            \
     86   };                                                                                                                \
     87 }
     88 
     89   
     90 EIGEN_GL_FUNC_DECLARATION       (glVertex)
     91 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,int,    2,2iv)
     92 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,short,  2,2sv)
     93 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,float,  2,2fv)
     94 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,double, 2,2dv)
     95 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,int,    3,3iv)
     96 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,short,  3,3sv)
     97 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,float,  3,3fv)
     98 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,double, 3,3dv)
     99 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,int,    4,4iv)
    100 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,short,  4,4sv)
    101 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,float,  4,4fv)
    102 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glVertex,double, 4,4dv)
    103 
    104 EIGEN_GL_FUNC_DECLARATION       (glTexCoord)
    105 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,int,    2,2iv)
    106 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,short,  2,2sv)
    107 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,float,  2,2fv)
    108 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,double, 2,2dv)
    109 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,int,    3,3iv)
    110 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,short,  3,3sv)
    111 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,float,  3,3fv)
    112 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,double, 3,3dv)
    113 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,int,    4,4iv)
    114 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,short,  4,4sv)
    115 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,float,  4,4fv)
    116 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTexCoord,double, 4,4dv)
    117 
    118 EIGEN_GL_FUNC_DECLARATION       (glColor)
    119 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,int,    2,2iv)
    120 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,short,  2,2sv)
    121 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,float,  2,2fv)
    122 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,double, 2,2dv)
    123 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,int,    3,3iv)
    124 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,short,  3,3sv)
    125 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,float,  3,3fv)
    126 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,double, 3,3dv)
    127 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,int,    4,4iv)
    128 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,short,  4,4sv)
    129 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,float,  4,4fv)
    130 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glColor,double, 4,4dv)
    131 
    132 EIGEN_GL_FUNC_DECLARATION       (glNormal)
    133 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glNormal,int,    3,3iv)
    134 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glNormal,short,  3,3sv)
    135 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glNormal,float,  3,3fv)
    136 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glNormal,double, 3,3dv)
    137 
    138 inline void glScale2fv(const float*  v) { glScalef(v[0], v[1], 1.f);  }
    139 inline void glScale2dv(const double* v) { glScaled(v[0], v[1], 1.0);  }
    140 inline void glScale3fv(const float*  v) { glScalef(v[0], v[1], v[2]); }
    141 inline void glScale3dv(const double* v) { glScaled(v[0], v[1], v[2]); }
    142 
    143 EIGEN_GL_FUNC_DECLARATION       (glScale)
    144 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glScale,float,  2,2fv)
    145 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glScale,double, 2,2dv)
    146 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glScale,float,  3,3fv)
    147 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glScale,double, 3,3dv)
    148 
    149 template<typename Scalar> void glScale(const UniformScaling<Scalar>& s)  { glScale(Matrix<Scalar,3,1>::Constant(s.factor())); }
    150 
    151 inline void glTranslate2fv(const float*  v) { glTranslatef(v[0], v[1], 0.f);  }
    152 inline void glTranslate2dv(const double* v) { glTranslated(v[0], v[1], 0.0);  }
    153 inline void glTranslate3fv(const float*  v) { glTranslatef(v[0], v[1], v[2]); }
    154 inline void glTranslate3dv(const double* v) { glTranslated(v[0], v[1], v[2]); }
    155 
    156 EIGEN_GL_FUNC_DECLARATION       (glTranslate)
    157 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTranslate,float,  2,2fv)
    158 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTranslate,double, 2,2dv)
    159 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTranslate,float,  3,3fv)
    160 EIGEN_GL_FUNC_SPECIALIZATION_VEC(glTranslate,double, 3,3dv)
    161 
    162 template<typename Scalar> void glTranslate(const Translation<Scalar,2>& t)  { glTranslate(t.vector()); }
    163 template<typename Scalar> void glTranslate(const Translation<Scalar,3>& t)  { glTranslate(t.vector()); }
    164 
    165 EIGEN_GL_FUNC_DECLARATION       (glMultMatrix)
    166 EIGEN_GL_FUNC_SPECIALIZATION_MAT(glMultMatrix,float,  4,4,f)
    167 EIGEN_GL_FUNC_SPECIALIZATION_MAT(glMultMatrix,double, 4,4,d)
    168 
    169 template<typename Scalar> void glMultMatrix(const Transform<Scalar,3,Affine>& t)        { glMultMatrix(t.matrix()); }
    170 template<typename Scalar> void glMultMatrix(const Transform<Scalar,3,Projective>& t)    { glMultMatrix(t.matrix()); }
    171 template<typename Scalar> void glMultMatrix(const Transform<Scalar,3,AffineCompact>& t) { glMultMatrix(Transform<Scalar,3,Affine>(t).matrix()); }
    172 
    173 EIGEN_GL_FUNC_DECLARATION       (glLoadMatrix)
    174 EIGEN_GL_FUNC_SPECIALIZATION_MAT(glLoadMatrix,float,  4,4,f)
    175 EIGEN_GL_FUNC_SPECIALIZATION_MAT(glLoadMatrix,double, 4,4,d)
    176 
    177 template<typename Scalar> void glLoadMatrix(const Transform<Scalar,3,Affine>& t)        { glLoadMatrix(t.matrix()); }
    178 template<typename Scalar> void glLoadMatrix(const Transform<Scalar,3,Projective>& t)    { glLoadMatrix(t.matrix()); }
    179 template<typename Scalar> void glLoadMatrix(const Transform<Scalar,3,AffineCompact>& t) { glLoadMatrix(Transform<Scalar,3,Affine>(t).matrix()); }
    180 
    181 inline void glRotate(const Rotation2D<float>& rot)
    182 {
    183   glRotatef(rot.angle()*180.f/float(M_PI), 0.f, 0.f, 1.f);
    184 }
    185 inline void glRotate(const Rotation2D<double>& rot)
    186 {
    187   glRotated(rot.angle()*180.0/M_PI, 0.0, 0.0, 1.0);
    188 }
    189 
    190 template<typename Derived> void glRotate(const RotationBase<Derived,3>& rot)
    191 {  
    192   Transform<typename Derived::Scalar,3,Projective> tr(rot);
    193   glMultMatrix(tr.matrix());
    194 }
    195 
    196 #define EIGEN_GL_MAKE_CONST_const const
    197 #define EIGEN_GL_MAKE_CONST__ 
    198 #define EIGEN_GL_EVAL(X) X
    199 
    200 #define EIGEN_GL_FUNC1_DECLARATION(FUNC,ARG1,CONST)                                                                             \
    201 namespace internal {                                                                                                            \
    202   template< typename XprType,                                                                                                   \
    203             typename Scalar = typename XprType::Scalar,                                                                         \
    204             int Rows = XprType::RowsAtCompileTime,                                                                              \
    205             int Cols = XprType::ColsAtCompileTime,                                                                              \
    206             bool IsGLCompatible = bool(XprType::Flags&LinearAccessBit)                                                          \
    207                               && bool(XprType::Flags&DirectAccessBit)                                                           \
    208                               && (XprType::IsVectorAtCompileTime || (XprType::Flags&RowMajorBit)==0)>                           \
    209   struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl);                                                                                  \
    210                                                                                                                                 \
    211   template<typename XprType, typename Scalar, int Rows, int Cols>                                                               \
    212   struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<XprType,Scalar,Rows,Cols,false> {                                                 \
    213     inline static void run(ARG1 a,EIGEN_GL_EVAL(EIGEN_GL_MAKE_CONST_##CONST) XprType& p) {                                      \
    214       EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<typename plain_matrix_type_column_major<XprType>::type>::run(a,p); }                 \
    215   };                                                                                                                            \
    216 }                                                                                                                               \
    217                                                                                                                                 \
    218 template<typename Derived> inline void FUNC(ARG1 a,EIGEN_GL_EVAL(EIGEN_GL_MAKE_CONST_##CONST) Eigen::DenseBase<Derived>& p) {   \
    219   EIGEN_CAT(EIGEN_CAT(internal::gl_,FUNC),_impl)<Derived>::run(a,p.derived());                                                  \
    220 }
    221 
    222 
    223 #define EIGEN_GL_FUNC1_SPECIALIZATION_MAT(FUNC,ARG1,CONST,SCALAR,ROWS,COLS,SUFFIX)                                              \
    224 namespace internal {                                                                                                            \
    225   template< typename XprType> struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<XprType, SCALAR, ROWS, COLS, true> {                  \
    226     inline static void run(ARG1 a, EIGEN_GL_EVAL(EIGEN_GL_MAKE_CONST_##CONST) XprType& p) { FUNC##SUFFIX(a,p.data()); }         \
    227   }; \
    228 }
    229 
    230   
    231 #define EIGEN_GL_FUNC1_SPECIALIZATION_VEC(FUNC,ARG1,CONST,SCALAR,SIZE,SUFFIX)                                                   \
    232 namespace internal {                                                                                                            \
    233   template< typename XprType> struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<XprType, SCALAR, SIZE, 1, true> {                     \
    234     inline static void run(ARG1 a, EIGEN_GL_EVAL(EIGEN_GL_MAKE_CONST_##CONST) XprType& p) { FUNC##SUFFIX(a,p.data()); }         \
    235   };                                                                                                                            \
    236   template< typename XprType> struct EIGEN_CAT(EIGEN_CAT(gl_,FUNC),_impl)<XprType, SCALAR, 1, SIZE, true> {                     \
    237     inline static void run(ARG1 a, EIGEN_GL_EVAL(EIGEN_GL_MAKE_CONST_##CONST) XprType& p) { FUNC##SUFFIX(a,p.data()); }         \
    238   };                                                                                                                            \
    239 }
    240 
    241 EIGEN_GL_FUNC1_DECLARATION       (glGet,GLenum,_)
    242 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glGet,GLenum,_,float,  4,4,Floatv)
    243 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glGet,GLenum,_,double, 4,4,Doublev)
    244 
    245 // glUniform API
    246 
    247 #ifdef GL_VERSION_2_0
    248 
    249 inline void glUniform2fv_ei  (GLint loc, const float* v)         { glUniform2fv(loc,1,v); }
    250 inline void glUniform2iv_ei  (GLint loc, const int* v)           { glUniform2iv(loc,1,v); }
    251 
    252 inline void glUniform3fv_ei  (GLint loc, const float* v)         { glUniform3fv(loc,1,v); }
    253 inline void glUniform3iv_ei  (GLint loc, const int* v)           { glUniform3iv(loc,1,v); }
    254 
    255 inline void glUniform4fv_ei  (GLint loc, const float* v)         { glUniform4fv(loc,1,v); }
    256 inline void glUniform4iv_ei  (GLint loc, const int* v)           { glUniform4iv(loc,1,v); }
    257 
    258 inline void glUniformMatrix2fv_ei  (GLint loc, const float* v)         { glUniformMatrix2fv(loc,1,false,v); }
    259 inline void glUniformMatrix3fv_ei  (GLint loc, const float* v)         { glUniformMatrix3fv(loc,1,false,v); }
    260 inline void glUniformMatrix4fv_ei  (GLint loc, const float* v)         { glUniformMatrix4fv(loc,1,false,v); }
    261 
    262 
    263 EIGEN_GL_FUNC1_DECLARATION       (glUniform,GLint,const)
    264 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,float,        2,2fv_ei)
    265 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,int,          2,2iv_ei)
    266 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,float,        3,3fv_ei)
    267 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,int,          3,3iv_ei)
    268 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,float,        4,4fv_ei)
    269 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,int,          4,4iv_ei)
    270 
    271 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glUniform,GLint,const,float,        2,2,Matrix2fv_ei)
    272 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glUniform,GLint,const,float,        3,3,Matrix3fv_ei)
    273 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glUniform,GLint,const,float,        4,4,Matrix4fv_ei)
    274 
    275 #endif
    276 
    277 #ifdef GL_VERSION_2_1
    278 
    279 static void glUniformMatrix2x3fv_ei(GLint loc, const float* v)         { glUniformMatrix2x3fv(loc,1,false,v); }
    280 static void glUniformMatrix3x2fv_ei(GLint loc, const float* v)         { glUniformMatrix3x2fv(loc,1,false,v); }
    281 static void glUniformMatrix2x4fv_ei(GLint loc, const float* v)         { glUniformMatrix2x4fv(loc,1,false,v); }
    282 static void glUniformMatrix4x2fv_ei(GLint loc, const float* v)         { glUniformMatrix4x2fv(loc,1,false,v); }
    283 static void glUniformMatrix3x4fv_ei(GLint loc, const float* v)         { glUniformMatrix3x4fv(loc,1,false,v); }
    284 static void glUniformMatrix4x3fv_ei(GLint loc, const float* v)         { glUniformMatrix4x3fv(loc,1,false,v); }
    285 
    286 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glUniform,GLint,const,float,        2,3,Matrix2x3fv_ei)
    287 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glUniform,GLint,const,float,        3,2,Matrix3x2fv_ei)
    288 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glUniform,GLint,const,float,        2,4,Matrix2x4fv_ei)
    289 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glUniform,GLint,const,float,        4,2,Matrix4x2fv_ei)
    290 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glUniform,GLint,const,float,        3,4,Matrix3x4fv_ei)
    291 EIGEN_GL_FUNC1_SPECIALIZATION_MAT(glUniform,GLint,const,float,        4,3,Matrix4x3fv_ei)
    292 
    293 #endif
    294 
    295 #ifdef GL_VERSION_3_0
    296 
    297 inline void glUniform2uiv_ei (GLint loc, const unsigned int* v)  { glUniform2uiv(loc,1,v); }
    298 inline void glUniform3uiv_ei (GLint loc, const unsigned int* v)  { glUniform3uiv(loc,1,v); }
    299 inline void glUniform4uiv_ei (GLint loc, const unsigned int* v)  { glUniform4uiv(loc,1,v); }
    300 
    301 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,unsigned int, 2,2uiv_ei)
    302 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,unsigned int, 3,3uiv_ei)
    303 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,unsigned int, 4,4uiv_ei)
    304 
    305 #endif
    306 
    307 #ifdef GL_ARB_gpu_shader_fp64
    308 inline void glUniform2dv_ei  (GLint loc, const double* v)        { glUniform2dv(loc,1,v); }
    309 inline void glUniform3dv_ei  (GLint loc, const double* v)        { glUniform3dv(loc,1,v); }
    310 inline void glUniform4dv_ei  (GLint loc, const double* v)        { glUniform4dv(loc,1,v); }
    311 
    312 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,double,       2,2dv_ei)
    313 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,double,       3,3dv_ei)
    314 EIGEN_GL_FUNC1_SPECIALIZATION_VEC(glUniform,GLint,const,double,       4,4dv_ei)
    315 #endif
    316 
    317 
    318 //@}
    319 
    320 }
    321 
    322 #endif // EIGEN_OPENGL_MODULE
    323