• Direct3D 11で深度バッファシャドウ

    2011年07月18日 22時24分
    MMDのシェーダを見ていると、深度バッファシャドウが入っているようなので入れてみます。

    一口に深度バッファシャドウと言っても細かい実装方法が色々ありますので、
    厳密にはMMDの処理とは違うと思います。
    深度バッファシャドウの一例としてご覧下さい。


    今回のソース


    深度バッファシャドウを実現するために必要になるのが
    Z値を保持するテクスチャです。これを生成します。

    ポイントはD3D11_TEXTURE2D_DESC.Usageです。
    Render targetとするためにD3D11_BIND_FLAG.D3D11_BIND_RENDER_TARGET
    Shader resourceとするためにD3D11_BIND_FLAG.D3D11_BIND_SHADER_RESOURCEを設定します。
    1D3D11_TEXTURE2D_DESC renderTextureDesc;
    2renderTextureDesc.Width = width;
    3renderTextureDesc.Height = height;
    4renderTextureDesc.MipLevels = 1;
    5renderTextureDesc.ArraySize = 1;
    6renderTextureDesc.Format = DXGI_FORMAT.DXGI_FORMAT_R32_FLOAT;
    7renderTextureDesc.SampleDesc.Count = 1;
    8renderTextureDesc.SampleDesc.Quality = 0;
    9renderTextureDesc.Usage = D3D11_USAGE.D3D11_USAGE_DEFAULT;
    10renderTextureDesc.BindFlags = D3D11_BIND_FLAG.D3D11_BIND_SHADER_RESOURCE | D3D11_BIND_FLAG.D3D11_BIND_RENDER_TARGET;
    11renderTextureDesc.CPUAccessFlags = 0;
    12if (FAILED(device.CreateTexture2D(&renderTextureDesc, null, &renderTarget_)))
    13{
    14    cleanup();
    15    throw new Error("Failed create depth stencil shadow texture");
    16}


    生成したテクスチャは、Z値を書き込む必要があるため、Render target viewを生成します。
    1if (FAILED(device.CreateRenderTargetView(renderTarget_, null, &renderTargetView_)))
    2{
    3    cleanup();
    4    throw new Error("Failed create depth stencil shadow texture view");
    5}



    また、Z値を参照する必要もあるため、Shader resource viewも生成します。
    1if (FAILED(device.CreateShaderResourceView(renderTarget_, null, &renderTargetResourceView_)))
    2{
    3    cleanup();
    4    throw new Error("Failed create depth stencil shadow texture resource view");
    5}





    このテクスチャにZ値を書き込むには、
    生成したRender target viewを使用してRender targetに設定します。
    1deviceContext.OMSetRenderTargets(1, &renderTargetView_, depthStencilView_);



    この状態でレンダリングします。

    ビュー変換は、ライトをカメラとして変換を行います。
    また、射影変換は正射影変換を使っています。

    そして、ピクセルシェーダの出力が色データではなくZ値になります。
    1return input.pos.z;



    この出力したZ値を色として見るとこうなります。
    深度データテクスチャ




    生成されたZ値テクスチャを、通常のレンダリングするときにピクセルシェーダにセットします。
    1immediateContext_.PSSetShaderResources(1, 1, &depthTextureView);



    頂点シェーダでは、Z値テクスチャを生成した時と同じワールドビュー射影変換も追加で行います。
    このXY座標は-1~1の値になっているので、0~1のテクスチャ座標に変換します。
    1float2 depthTexCoord;
    2depthTexCoord.x = (input.lpos.x + 1) * 0.5;
    3depthTexCoord.y = (1 - input.lpos.y) * 0.5;


    この座標を使ってテクスチャからデータを取ると、深度データが取れます。
    1float minDepth = depthTex.Sample(sam, depthTexCoord).r + 0.0005f;


    取得したテクスチャのZ値と、そのピクセルのZ値(Z値テクスチャを生成した時と同じワールドビュー射影変換した座標)を比較し、
    テクスチャのZ値より大きければ、ライトの処理をキャンセルします。(ライトが遮られた状態と判断出来る)


    結果、このように影が表示されます。
    深度バッファシャドウ適用



    パイプラインの入力と出力には同じデータがセット出来ません。
    次にRender targetにテクスチャをセットするために、ピクセルシェーダにセットしたテクスチャは外しておきます。
    1ID3D11ShaderResourceView nullView = null;
    2immediateContext_.PSSetShaderResources(1, 1, &nullView);


    Render tagetも同様に外す必要がありますが、
    今回の場合、バックバッファをRender targetにした時点で外れるので、明示的には外していません。


    Z値テクスチャを生成するときと、通常のレンダリングをするとき、
    どちらもパイプラインの設定はほとんど同じになるので、エフェクトファイルを使うべきかなぁとも思います。
    手動だとすごく面倒です。
  • D言語でDirect3D 11を使う時のデバッグ出力参照

    2011年07月18日 13時12分
    正確には、Dに限らずVisualStudioのようなIDEが無くてもデバッグ出力を見る方法です。


    まずはアプリケーション側のデバッグ出力有効化です。
    この作業は表示側がIDEでも他のでも共通です。


    デバッグ出力を有効にするには、D3D11CreateDeviceD3D11CreateDeviceAndSwapChainを呼び出すときに
    デバッグフラグを指定してやります。

    具体的には、FlagsにD3D11_CREATE_DEVICE_FLAG.D3D11_CREATE_DEVICE_DEBUGを指定します。


    このフラグを有効にするためには、D3D11SDKLayers.dllが必要になるので、
    DirectXのSDKをインストールする必要があります。


    詳しくは公式を参照下さい。
    ソフトウェアレイヤー



    これでアプリケーション側は出力するようになったので、
    何かしらの方法でキャプチャすれば見ることが出来ます。

    DirectXのデバッグ出力は、
    標準入力等ではなくWin32のOutputDebugStringという仕組みを使っているようです。

    デバッグ用の仕組みなので、当然キャプチャするアプリが準備されています。
    DebugView for Windows


    使い方はDebugViewを起動した状態で、
    OutputDebugStringを使うアプリを起動するだけです。
  • Direct3D 11のアルファテスト

    2011年07月16日 12時25分
    前回の続き。

    ここまで来ればあとは簡単です。

    肌が表示されていた部分が透明になっているということは、
    透明ピクセルの時もZバッファへの書き込みがされているということです。

    アルファテストを入れて、透明ピクセルの時はZバッファに書き込まないようにします。


    ただ、Direct3D 11にはアルファテストという機能自体はありません。
    ピクセルシェーダで代わりに処理をしてあげる必要があります。



    通常、ピクセルシェーダは最後にピクセル情報を返します。

    例としてはこんな感じ
    1return float4(ambient + diffuse + specular, alpha);



    ここで、ピクセルシェーダでのみ使用出来る制御構文discardを使用します。
    1// アルファ値0の場合は表示しない
    2if (alpha == 0)
    3    discard;


    discardを使用するとピクセルを出力しないため、Zバッファへの書き込みもキャンセルすることが出来ます。


    アルファ値0の時にdiscardをするように処理すると、正常に表示されるようになりました。
    discard使用



    大体動いた気がするので、現時点のソースを置いておきます。参考になるかな。
  • Direct3D 11のアルファブレンド設定

    2011年07月15日 22時43分
    前回の続き。

    ニーソ破れる問題がどうしてもわからず悩み続けました。
    悩みすぎて夢の中でもコードを書いている状態です。

    しかし、それ程苦しんだ甲斐があり、閃きが突然訪れました。




    「ぱんつのテクスチャ透明にすれば、ぱんつ脱げるんじゃね!?」




    はやる気持ちを抑え、ボクはテクスチャを編集しました。
    幸いテクスチャは解りやすく、容易くぱんつの部分を透明にすることに成功しました。

    圧倒的な達成感に包まれながらボクはプログラムでモデルを表示しました。


    しかし、待っていたのは思わぬ結果でした。
    なんの変化もありません。


    その後、アルファ値の設定を変えたりしてみて気づいたのですが、
    アルファブレンドの処理がされていないようです。

    そこで今回はアルファブレンドの設定方法です。

    ステータス設定はいつもの手順です。
    1. D3D11_BLEND_DESCにアルファブレンドの設定を入れる
    2. D3D11_BLEND_DESCをID3D11Device.CreateBlendStateに渡してID3D11BlendStateを生成
    3. ID3D11DeviceContext.OMSetBlendStateにID3D11BlendStateをセット
    4. ID3D11BlendStateは終了時にReleaseを忘れずに


    ちょっとはまった点。

    RenderTargetのSrcBlendとDestBlendはD3D11_BLEND_SRC_COLORとかを入れたくるのですが、
    通常はD3D11_BLEND_SRC_ALPHA等を入れるのが正解です。
    これらはRGBのブレンド設定ですが、ブレンドの係数自体はアルファ値を使うためです。


    アルファ値を設定して描画です。
    アルファ値設定済み


    体と足が表示されない!

    つまり、元々このモデルデータは体と足が透明になるのが正しいわけです。
    ということはこの状態で服も描画すれば正常な表示なる!


    アルファ値設定済み全部描画


    ならなかった。
  • Direct3D 11のZFUNC設定

    2011年07月15日 00時02分
    前回の続き。

    さらに表示を直して行きます。

    現在はニーソが破れたような感じになっています。
    また、よく見ると上着も穴が空いています。


    理由がよく解らないので、原因を切り分けようと服と体を別々に表示してみました。

    服のみを表示
    服のみ

    体のみを表示
    体のみ


    どちらも正常に表示されています。
    ということは、Z値が正常でないため、前後関係がおかしくなっていると予想されます。

    他の部分についてはZ値が正常に働いているように見えるので
    ほんのわずかな誤差のせいではないでしょうか。

    というわけで、試しにZバッファの精度を落としてみました。

    現在は32bitにしているところを
    1textureDesc.Format = DXGI_FORMAT.DXGI_FORMAT_D32_FLOAT;


    16bitにしてみます。
    1textureDesc.Format = DXGI_FORMAT.DXGI_FORMAT_D16_UNORM;


    微妙に破れた範囲が広がりました。


    あと思いつくのはZFUNCでしょうか。
    ちょっと比較式を変更してみましょう。

    Direct3D 11のZFUNCの設定ですが、例によってSetRenderStateを呼び出してはいオッケーでは無くなっています。
    次の手順が必要です。
    1. D3D11_DEPTH_STENCIL_DESCに深度ステンシルの設定を入れる
    2. D3D11_DEPTH_STENCIL_DESCをID3D11Device.CreateDepthStencilStateに渡してID3D11DepthStencilStateを生成
    3. ID3D11DeviceContext.OMSetDepthStencilStateにID3D11DepthStencilStateをセット
    4. ID3D11DepthStencilStateは終了時にReleaseを忘れずに

    また面倒ですが、このパターンにも慣れてきました。


    これで比較式を変更出来るので、色々な設定で試してみましたが全く改善されません。
    どうもZバッファの問題ではなさそうです。