我正在制作在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的
答案 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
和一些元编程提供一些有限形式的概念检查。