Používání modelu C++ AMP v aplikacích pro Windows Store
C++ AMP (C++ Accelerated Massive Parallelism) můžete použít ve vaší aplikaci Windows Store pro výpočty na GPU (Graphic Processing Unit) nebo jiných počítačových akcelerátorech.Nicméně C++ AMP neposkytuje rozhraní API pro práci přímo s typy prostředí Windows Runtime a prostředí Windows Runtime nenabízí obal pro C++ AMP.Při použití typů Windows Runtime ve vašem kódu, včetně těch, které jste sami vytvořili, je nutné je převést na typy, které jsou kompatibilní s C++ AMP.
Důležité informace o výkonu
Pokud používáte rozšíření komponent v jazyce Visual C++ (C++/CX) k vytvoření své aplikace Windows Store, doporučujeme použít staré jednoduché typy (POD) spolu se souvislým úložištěm – například std::vector nebo pole stylu jazyka C – pro data, která budou používána v C++ AMP.To vám může pomoci dosáhnout vyššího výkonu než pomocí typů bez POD nebo kontejnerů RT systému Windows, protože nemusí dojít k žádnému zařazování.
V jádře C++ AMP pro přístup k datům uloženým tímto způsobem pouze zabalte std::vector nebo pole úložiště do concurrency::array_view a potom použijte zobrazení pole ve smyčce concurrency::parallel_for_each:
// simple vector addition example
std::vector<int> data0(1024, 1);
std::vector<int> data1(1024, 2);
std::vector<int> data_out(data0.size(), 0);
concurrency::array_view<int, 1> av0(data0.size(), data0);
concurrency::array_view<int, 1> av1(data1.size(), data1);
concurrency::array_view<int, 1> av2(data_out.size(), data2);
av2.discard_data();
concurrency::parallel_for_each(av0.extent, [=](concurrency::index<1> idx) restrict(amp)
{
av2[idx] = av0[idx] + av1[idx];
});
Zařazování typů modulu Windows Runtime
Při práci s rozhraními API Windows Runtime můžete chtít použít C++ AMP na data uložená v kontejneru Windows Runtime, jako například Platform::Array<T>^, nebo v komplexních datových typech, jako jsou třídy nebo struktury, které jsou deklarovány pomocí klíčového slova ref nebovalue.V těchto situacích je nutné provést další úkony a zpřístupnit tak data v rámci C++ AMP.
Platform::Array<T>^, kde T je typ POD
Pokud se setkáte s Platform::Array<T>^ a T je typ POD, k jeho základnímu úložišti můžete přistupovat jednoduše pomocí členské funkce get:
Platform::Array<float>^ arr; // Assume that this was returned by a Windows Runtime API
concurrency::array_view<float, 1> av(arr->Length, &arr->get(0));
Pokud T není POD typ, použijte techniku, která je popsána v následující části pro použití dat s C++ AMP.
Typy modulu Windows Runtime: třídy deklarované s použitím klíčových slov ref a value
C++ AMP nepodporuje komplexní datové typy.Jedná se o typy bez POD a všechny typy, které jsou deklarovány pomocí klíčového slova ref nebo value.Pokud je použit nepodporovaný typ v kontext restrict(amp), chyba kompilace je vygenerována.
Když narazíte na nepodporovaný typ, můžete kopírovat zajímavé části data do objektu concurrency::array.Kromě zpřístupnění dat pro použití C++ AMP, může tento postup ručního kopírování také zvýšit výkon maximalizací lokality dat a zajištěním, že data, která nebudou použita, nejsou zkopírována do akcelerátoru.Dalšího zlepšení výkonu dosáhnete použitím pracovního pole, což je zvláštní forma concurrency::array poskytující nápovědu k modulu runtime AMP, že pole má být optimalizováno pro častý přenos mezi ním a jinými poli na určený akcelerátor.
// pixel_color.h
ref class pixel_color sealed
{
public:
pixel_color(Platform::String^ color_name, int red, int green, int blue)
{
name = color_name;
r = red;
g = green;
b = blue;
}
property Platform::String^ name;
property int r;
property int g;
..property int b;
};
// Some other file
std::vector<pixel_color^> pixels (256);
for(pixel_color ^pixel : pixels)
{
pixels.push_back(ref new pixel_color("blue", 0, 0, 255));
}
// Create the accelerators
auto cpuAccelerator = concurrency::accelerator(concurrency::accelerator::cpu_accelerator);
auto devAccelerator = concurrency::accelerator(concurrency::accelerator::default_accelerator);
// Create the staging arrays
concurrency::array<float, 1> red_vec(256, cpuAccelerator.default_view, devAccelerator.default_view);
concurrency::array<float, 1> blue_vec(256, cpuAccelerator.default_view, devAccelerator.default_view);
// Extract data from the complex array of structs into staging arrays.
concurrency::parallel_for(0, 256, [&](int i)
{
red_vec[i] = pixels[i]->r;
blue_vec[i] = pixels[i]->b;
});
// Array views are still used to copy data to the accelerator
concurrency::array_view<float, 1> av_red(red_vec);
concurrency::array_view<float, 1> av_blue(blue_vec);
// Change all pixels from blue to red.
concurrency::parallel_for_each(av_red.extent, [=](index<1> idx) restrict(amp)
{
av_red[idx] = 255;
av_blue[idx] = 0;
});