
एकता के साथ कम्प्यूटेशनल शेड के उपयोग के बारे में पहले से ही हैबर पर बहुत सारे लेख हैं , हालांकि, "क्लीन" Win32 एपीआई + डायरेक्टएक्स 11 पर कम्प्यूटेशनल शेडर का उपयोग करने के बारे में एक लेख खोजना मुश्किल है। हालांकि, यह कार्य अधिक जटिल नहीं है, अधिक विवरण में - कटौती के तहत।
ऐसा करने के लिए, हम उपयोग करेंगे:
- विंडोज 10
- विजुअल स्टूडियो 2017 सामुदायिक संस्करण "C ++ में क्लासिक एप्लिकेशन का विकास" मॉड्यूल के साथ
प्रोजेक्ट बनाने के बाद, हम लिंकर को `d3d11.lib` लाइब्रेरी का उपयोग करने के लिए कहेंगे।

हैडर फाइलेंप्रति सेकंड फ़्रेम की संख्या की गणना करने के लिए, हम मानक पुस्तकालय का उपयोग करेंगे
#include <time.h>
हम विंडो शीर्षक के माध्यम से प्रति सेकंड फ़्रेम की संख्या को आउटपुट करेंगे, जिसके लिए हमें संबंधित लाइन बनाने की आवश्यकता होगी
#include <stdio.h>
हम त्रुटि से निपटने पर विस्तार से विचार नहीं करेंगे, हमारे मामले में यह पर्याप्त है कि एप्लिकेशन डिबग संस्करण में क्रैश हो जाता है और गिरावट के समय इंगित करता है:
#include <assert.h>
WinAPI के लिए हैडर फाइलें:
#define WIN32_LEAN_AND_MEAN #include <tchar.h> #include <Windows.h>
Direct3D 11 के लिए हैडर फाइलें:
#include <dxgi.h> #include <d3d11.h>
शेडर लोड करने के लिए संसाधन आईडी। इसके बजाय, आप HLSL कंपाइलर द्वारा उत्पन्न शैडर ऑब्जेक्ट फ़ाइल को मेमोरी में लोड कर सकते हैं। संसाधन फ़ाइल बनाना बाद में वर्णित किया गया है।
#include "resource.h"
शॉल्डर और कॉलिंग हिस्से के लिए कॉन्सटेंट को एक अलग हेडर फ़ाइल में घोषित किया जाएगा।
#include "SharedConst.h"
हम विंडोज घटनाओं के प्रसंस्करण के लिए एक समारोह की घोषणा करते हैं, जिसे बाद में परिभाषित किया जाएगा:
LRESULT CALLBACK WndProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam);
हम एक विंडो बनाने और नष्ट करने के लिए फ़ंक्शन लिखेंगे
int windowWidth, windowHeight; HINSTANCE hInstance; HWND hWnd; void InitWindows() {
अगला वीडियो कार्ड (डिवाइस और डिवाइस कॉनटेक्स्ट) और आउटपुट बफ़र्स (स्वैचेचिन) की श्रृंखला तक पहुँचने के लिए इंटरफेस की शुरूआत है:
IDXGISwapChain *swapChain; ID3D11Device *device; ID3D11DeviceContext *deviceContext; void InitSwapChain() { HRESULT result; DXGI_SWAP_CHAIN_DESC swapChainDesc;
बफर से लेकर रेंडर तक पहुंच का आरंभिक प्रदर्शन किया जाएगा:
ID3D11RenderTargetView *renderTargetView; void InitRenderTargetView() { HRESULT result; ID3D11Texture2D *backBuffer;
शेड्स को शुरू करने से पहले, आपको उन्हें बनाने की आवश्यकता है। Visual Studio फ़ाइल एक्सटेंशन को पहचान सकता है, इसलिए हम केवल एक्सटेंशन .hlsl
साथ एक स्रोत बना सकते हैं, या सीधे मेनू के माध्यम से एक shader बना सकते हैं। मैंने पहला तरीका चुना, क्योंकि वैसे भी, गुणों के माध्यम से आपको Shader Model 5 का उपयोग सेट करना होगा।


इसी तरह, वर्टेक्स और पिक्सेल शेड्स बनाएं।
वर्टियर शेडर में, हम निर्देशांक को दो-आयामी वेक्टर से परिवर्तित करते हैं (क्योंकि हमारे पास जो बिंदु हैं वे दो आयामी हैं) को चार-आयामी (वीडियो कार्ड द्वारा प्राप्त):
float4 main(float2 input: POSITION): SV_POSITION { return float4(input, 0, 1); }
पिक्सेल शेडर में, हम सफेद वापस आएंगे:
float4 main(float4 input: SV_POSITION): SV_TARGET { return float4(1, 1, 1, 1); }
अब एक कम्प्यूटेशनल shader। हम इस सूत्र को बिंदुओं की परस्पर क्रिया के लिए परिभाषित करते हैं:
द्रव्यमान के साथ अपनाया 1
इस तरह से एचएलएसएल पर इसका कार्यान्वयन दिखेगा:
#include "SharedConst.h"
आप देख सकते हैं कि SharedConst.h
फ़ाइल shader में शामिल है। यह स्थिरांक के साथ हेडर फ़ाइल है, जो main.cpp
में शामिल है। यहाँ इस फ़ाइल की सामग्री है:
#ifndef PARTICLE_COUNT #define PARTICLE_COUNT (1 << 15) #endif #ifndef NUMTHREADS #define NUMTHREADS 64 #endif
बस एक समूह में कणों की संख्या और धाराओं की संख्या की घोषणा करना। हम प्रत्येक कण को एक धारा आवंटित करेंगे, इसलिए हम PARTICLE_COUNT / NUMTHREADS
रूप में समूहों PARTICLE_COUNT / NUMTHREADS
संख्या PARTICLE_COUNT / NUMTHREADS
करेंगे। यह संख्या पूर्णांक होनी चाहिए, इसलिए यह आवश्यक है कि कणों की संख्या समूह में प्रवाह की संख्या से विभाजित हो।
हम विंडोज संसाधन तंत्र का उपयोग करके संकलित shader bytecode को लोड करेंगे। ऐसा करने के लिए, निम्न फ़ाइलें बनाएँ:
resource.h
, जिसमें संबंधित संसाधन की ID होगी:
#pragma once #define IDR_BYTECODE_COMPUTE 101 #define IDR_BYTECODE_VERTEX 102 #define IDR_BYTECODE_PIXEL 103
और resource.rc
, निम्नलिखित सामग्री के संगत संसाधन उत्पन्न करने के लिए एक फ़ाइल:
#include "resource.h" IDR_BYTECODE_COMPUTE ShaderObject "compute.cso" IDR_BYTECODE_VERTEX ShaderObject "vertex.cso" IDR_BYTECODE_PIXEL ShaderObject "pixel.cso"
जहां ShaderObject
संसाधन का प्रकार है, और compute.cso
, vertex.cso
और pixel.cso
आउटपुट निर्देशिका में संकलित शेडर ऑब्जेक्ट फ़ाइलों के संबंधित नाम हैं।
फ़ाइलों को खोजने के लिए, आपको resource.rc
आउटपुट निर्देशिका में प्रोजेक्ट आउटपुट निर्देशिका के लिए पथ निर्दिष्ट करना चाहिए:

विज़ुअल स्टूडियो ने स्वचालित रूप से फ़ाइल को संसाधनों के विवरण के रूप में मान्यता दी और इसे विधानसभा में जोड़ा, आपको मैन्युअल रूप से ऐसा करने की आवश्यकता नहीं है
अब आप shader आरंभीकरण कोड लिख सकते हैं:
ID3D11ComputeShader *computeShader; ID3D11VertexShader *vertexShader; ID3D11PixelShader *pixelShader; ID3D11InputLayout *inputLayout; void InitShaders() { HRESULT result; HRSRC src; HGLOBAL res;
बफर इनिशियलाइज़ेशन कोड:
ID3D11Buffer *positionBuffer; ID3D11Buffer *velocityBuffer; void InitBuffers() { HRESULT result; float *data = new float[2 * PARTICLE_COUNT];
और कम्प्यूटेशनल shader से बफर एक्सेस आरंभीकरण कोड:
ID3D11UnorderedAccessView *positionUAV; ID3D11UnorderedAccessView *velocityUAV; void InitUAV() { HRESULT result;
अगला, आपको ड्राइवर को बफ़र्स के साथ बनाए गए शेड और बंडलों का उपयोग करना चाहिए:
void InitBindings() {
औसत फ्रेम समय की गणना करने के लिए, हम निम्नलिखित कोड का उपयोग करेंगे:
const int FRAME_TIME_COUNT = 128; clock_t frameTime[FRAME_TIME_COUNT]; int currentFrame = 0; float AverageFrameTime() { frameTime[currentFrame] = clock(); int nextFrame = (currentFrame + 1) % FRAME_TIME_COUNT; clock_t delta = frameTime[currentFrame] - frameTime[nextFrame]; currentFrame = nextFrame; return (float)delta / CLOCKS_PER_SEC / FRAME_TIME_COUNT; }
और प्रत्येक फ्रेम पर - इस फ़ंक्शन को कॉल करें:
void Frame() { float frameTime = AverageFrameTime();
यदि विंडो का आकार बदल गया है, तो हमें रेंडरिंग बफ़र्स के आकार को भी बदलना होगा:
void ResizeSwapChain() { HRESULT result; RECT rect;
अंत में, आप एक संदेश संसाधन फ़ंक्शन को परिभाषित कर सकते हैं:
LRESULT CALLBACK WndProc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam) { switch (Msg) { case WM_CLOSE: PostQuitMessage(0); break; case WM_KEYDOWN: if (wParam == VK_ESCAPE) PostQuitMessage(0); break; case WM_SIZE: ResizeSwapChain(); break; default: return DefWindowProc(hWnd, Msg, wParam, lParam); } return 0; }
और main
कार्य:
int main() { InitWindows(); InitSwapChain(); InitRenderTargetView(); InitShaders(); InitBuffers(); InitUAV(); InitBindings(); ShowWindow(hWnd, SW_SHOW); bool shouldExit = false; while (!shouldExit) { Frame(); MSG msg; while (!shouldExit && PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) { TranslateMessage(&msg); DispatchMessage(&msg); if (msg.message == WM_QUIT) shouldExit = true; } } DisposeUAV(); DisposeBuffers(); DisposeShaders(); DisposeRenderTargetView(); DisposeSwapChain(); DisposeWindows(); }
लेख के शीर्षक में रनिंग प्रोग्राम का स्क्रीनशॉट देखा जा सकता है।
→ प्रोजेक्ट पूरी तरह से GitHub में अपलोड किया गया है