定义使用此类型作为模板参数的类型和函数

时间:2014-02-09 15:48:32

标签: c++

我正在制作在CPU上生成mip贴图的功能。 我的目标是编写适用于许多DXGI_FORMAT的通用功能。函数对所有DXGI_FORMAT组合都做同样的事情,唯一的差异是数据类型和XMMATH函数来加载和存储它。

我尝试以很好的方式做到了 - 通过模板参数,例如:

#define AVERAGE(p, q) (((p - q) * 0.5f) + q)

template<typename XMATH_PIXEL_FORMAT,
         typename XMVECTOR(*XMATH_LOAD_FUNC)(const XMATH_PIXEL_FORMAT*),
         typename void(*XMATH_STORE_FUNC)(XMATH_PIXEL_FORMAT*, XMVECTOR)>
void minifyMipMap(XMATH_PIXEL_FORMAT* dst, const XMATH_PIXEL_FORMAT* src, size_t src_x, size_t src_y) {
  size_t x, y, x2, y2, dst_x = src_x >> 1, dst_y = src_y >> 1;
  XMVECTOR out, p00, p10, p01, p11; // p:xy

  for(y = 0; y < dst_y; ++y) { // for each dst line
    y2 = y << 1; // y2 = y * 2
    for(x = 0; x < dst_x; ++x) { // for each dst pixel
      x2 = x << 1; // x2 = x * 2

      // load 4 pixels
      p00 = XMATH_LOAD_FUNC(src + y2 * src_x + x2);
      p10 = XMATH_LOAD_FUNC(src + y2 * src_x + x2 + 1);
      p01 = XMATH_LOAD_FUNC(src + (y2 + 1) * src_x + x2);
      p11 = XMATH_LOAD_FUNC(src + (y2 + 1) * src_x + x2 + 1);

      // blend 4 pixels into 1
      out = AVERAGE(AVERAGE(p00, p10), AVERAGE(p01, p11));

      // store dest pixel
      XMATH_STORE_FUNC(dst + y * dst_x + x, out);
    }
  }
}


void test() {
  BYTE src[] = {
    0, 0, 0, 255, // black
    255, 255, 255, 255, // white
    0, 0, 0, 255, // black
    255, 255, 255, 255 // white
  };
  XMUBYTE4 dst[1]; // gray ???

  minifyMipMap<XMUBYTE4, XMLoadUByte4, XMStoreUByte4>(dst, (const XMUBYTE4*)src, 2, 2); // 2x2 = > 1x1
}

msvc说:C2893:无法专门化函数模板'void minifyMipMap(XMATH_PIXEL_FORMAT *,const XMATH_PIXEL_FORMAT *,size_t,size_t)' 使用以下模板参数: '的DirectX :: PackedVector :: XMUBYTE4'

我尝试用几种组合编译这段代码,但没有成功(我也尝试将类型定义为类模板,并将函数作为此类的静态方法参数,但它也失败了)。 我希望以某种方式在C ++中以很好的方式编写它 - 不使用#define的

1 个答案:

答案 0 :(得分:2)

如果您想接受任意功能,请不要这样做。每个不同的函数接受一个模板参数。

template<typename Func>
void foo(Func f) {
  // call f in any way you need
  f();
}

// can be used like this with functions
void bla() {}
f(&bla);
// Functors
struct X { void operator()() {} }
f(X());

如果传递的函数无法以代码调用的方式调用,编译器会抱怨。

如果您希望对用户真正有所帮助,可以使用static_assert和一些元编程提供一些有限形式的概念检查。