DirectX11, プログラム

D3DPreprocessでプリプロセス

1件のコメント

スクリプトやテキストデータを記述する際、Cプリプロセッサの機能があればかなり便利になります。しかし、それを実装するのは大変です。コンパイラ付属のプリプロセッサは入出力がファイルなので使い勝手がわるい。何かないか探していたところDirectxのシェーダコンパイラ関連の関数にD3DPreprocessというものを発見、使ってみる。

【送料無料】 225/45R18 18インチ ENKEI エンケイ PF01 7J 7.00-18 YOKOHAMA ヨコハマ ブルーアース RV-02 SALE サマータイヤ ホイール4本セット【YOsum18】が激安価格出販売中

D3DPreprocess(MSDN)
1
2
3
4
5
6
7
8
9
HRESULT D3DPreprocess(
  in   LPCVOID pSrcData,
  in   SIZE_T SrcDataSize,
  in   LPCSTR pSourceName,
  in   const D3D_SHADER_MACRO *pDefines,
  in   ID3DInclude pInclude,
  out  ID3DBlob *ppCodeText,
  out  ID3DBlob *ppErrorMsgs
);

HLSLコード用のプリプロセッサですが、スクリプトなどにも使えそうです。ファイルではなくデータ(メモリ)を指定します エスペリア Espelir スーパーダウンサス(フロント) Super DOWNSUS ホンダ N-BOXプラスカスタム JF2 H24/7~ マナレイ ユーロスピードS810 1本販売 ホイール エスクァイア【15×6J 5-114 INSET52】80系 MANARAY SPORT EuroSpeed マナレイスポーツ MiD アルミホイール 1枚 単品【店頭受取対応商品】。引数には処理するデータのほかに、ID3DBlobで結果受け取り、/Dオプションに相当するD3D_SHADER_MACRO、そして#include処理を行うインターフェイスを指定します。

include処理のインターフェイス定義

#includeでのファイル読み込み処理を実装するインターフェイス(クラス)を指定します。プリプロセッサオプション/Iに相当する機能はD3DPreprocessにないためこのクラスで実装します。D3D_COMPILE_STANDARD_FILE_INCLUDEを指定するとデフォルトのinclude処理(カレントディレクトリとpSourceNameのディレクトリの検索)。

プリプロセス結果
コメント除去、include展開、define置き換えが行われます。それとプリプロセス前の行数とincludeしたファイル名がわかるように#lineが追加されます。スクリプトの解析などでエラー発生場所を特定できます(プリプロセス後の行数では発生場所がわからない)。
1
2
3
4
#line 6 "test.h"
 
lineからの行数でプリプロセス前のファイルと行数がわかる
ここはtest.hの8行目

サンプルコード

includeインターフェイス
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
//D3DPreprocessでのinclude処理インターフェイス
// 検索パスを3つまで追加可能
class IInclude : public ID3DInclude
{
public:
    IInclude(const char* path0, const char* path1=nullptr, const char* path2=nullptr)
    {
        inc_path[0] = path0;
        inc_path[1] = path1;
        inc_path[2] = path2;
    }
    HRESULT __stdcall Open(D3D_INCLUDE_TYPE IncludeType, LPCSTR pFileName, LPCVOID pParentData, LPCVOID *ppData, UINT *pBytes)
    {
        switch(IncludeType) {
        // とりあえず""と<>の区別なし
        case D3D_INCLUDE_LOCAL://#include ""
        case D3D_INCLUDE_SYSTEM://#include <>
            break;
        default:
            return E_FAIL;
        }
 
        for(auto& p : inc_path){
            if(!p)continue;
            std::string path = p;
            path += pFileName;
 
            std::ifstream input;
            input.open(path.c_str(), std::ios::binary);
            if(!input.is_open())continue;
            size_t fsize = (size_t)input.seekg(0, std::ios::end).tellg();
            input.seekg(0, std::ios::beg);
 
            void* data = ::operator new(fsize);
            input.read(static_cast<char*>(data), fsize);
            *ppData = data;
            *pBytes = fsize;
            return S_OK;
        }
        return E_FAIL;
    }
 
    HRESULT __stdcall Close(LPCVOID pData)
    {
        ::operator delete(const_cast<void*>(pData));
        return S_OK;
    }
 
    const char* inc_path[3];//検索パス
};

使用例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
    const char* file_path = "data/test.fx";
    const char* inc_path = "data/";
 
    std::ifstream input;
    input.open("data/test.fx", std::ios::binary);
    //if(input.is_open())//省略
    size_t fsize = (size_t)input.seekg(0, std::ios::end).tellg();
    input.seekg(0, std::ios::beg);
    void* data = ::operator new(fsize);
    input.read(static_cast<char*>(data), fsize);
 
    IInclude iinc(inc_path);// /I inc_path
 
    // Cプリプロセッサでの/Dオプション
    D3D_SHADER_MACRO defines[] = {
        {"DEBUG", "1"},// /DDEBUG=1
        {nullptr, nullptr},// I'll be back
    };
 
    ID3DBlob* result = nullptr;
    ID3DBlob* error = nullptr;
    D3DPreprocess(data, fsize, file_path, defines, &iinc, &result, &error);
 
    if(result)OutputDebugStringA( (const char*)result->GetBufferPointer() );
    if(error)OutputDebugStringA( (const char*)error->GetBufferPointer() );
 
    ::operator delete(data);
    if(result)result->Release();
    if(error)error->Release();
 
// コードの色付けのバグにより ¥¥⇒/

おまけ

OutputDebugStringA関数などでエラー出力を行う場合
ファイルパス(数字):エラー文
という書式で出力すると、ダブルクリックでファイルパスのファイルをVisualStudioのエディタで開き、数字で指定した行数にエディットカーソルが移動した状態になります 。

TOYOTIRES トーヨー プロクセス CF2 SUV PROXES サマータイヤ 205/70R15 WEDS ジョーカーマジック ホイールセット 4本 15インチ 15 X 6 +53 5穴 114.3

【送料無料】 225/45R18 18インチ ENKEI エンケイ PF01 7J 7.00-18 YOKOHAMA ヨコハマ ブルーアース RV-02 SALE サマータイヤ ホイール4本セット【YOsum18】公式アウトレット


ファイルパス(10,8)
とすると10行、8文字目にカーソルが移動 16インチ サマータイヤ セット【適応車種:ノア(60系 全グレード)】HOT STUFF ヴァーレン W03 ガンメタリック 6.5Jx16LEMANS V LM5 205/55R16。
プリプロセスでエラーが発生したときに、ダブルクリックでエラー発生場所に移動できたので何か特殊なデバッグ機能がD3DPreprocessに備わっているのではと考えましたが、単にデバッグ出力を特定の書式にすることで有効になる機能でした。スクリプトやテキストデータの解析時に使えそうです 17インチ サマータイヤ セット【適応車種:クラウンロイヤル(180系 全グレード)】HOT STUFF エクシーダー E03 ダークシルバー 7.0Jx17ディレッツァ DZ102 215/55R17 [3518064] DIXCEL PD ブレーキローター フロント用 テルスター GFEPF/GFERF 97/10~99/9。