DirectX11, プログラム

D3DPreprocessでプリプロセス

1件のコメント

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

レースチップ XLRコネクト スロットルコントローラ RENAULT Megane III (2008 - 2016) エンジン1.6 16V Bifuel【RaceChip XLR Connect】品番1117【特価販売中!】

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コード用のプリプロセッサですが、スクリプトなどにも使えそうです。ファイルではなくデータ(メモリ)を指定します 【プロジェクトミュー】TYPE HC-CS(タイプエイチシー・シーエス) ニッサン セドリック CEDRIC用 YPY31系 リヤブレーキパッド 品番:R231 トヨタ カリブ AE115G ダウンサス RS-R DOWN T601W RSR☆ 新品 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のエディタで開き、数字で指定した行数にエディットカーソルが移動した状態になります 。

ブリッドパーツ ■BRIDE【品番:T709RO】 BRIDE シートレール ROタイプ

レースチップ XLRコネクト スロットルコントローラ RENAULT Megane III (2008 - 2016) エンジン1.6 16V Bifuel【RaceChip XLR Connect】品番1117









No車名エンジン排気量馬力トルクXLR適合品番
1Megane III (2008 - 2016)1.6 16V Bifuel15981101511117

注意
■車種別専用となりますので、適合品番を必ずご確認の上ご注文頂きます様お願い致します。
■同メーカーであっても車種により本体内部のプログラム及びハーネスコネクターが異なる事がございまので、必ずご購入品番と車種をご確認いただきます様お願い致します。





ファイルパス(10,8)
とすると10行、8文字目にカーソルが移動 【T236】bB [H12/2-H17/11][NCP35] カジュアルエスライン ホワイト Bellezza ベレッツァ シートカバー。
プリプロセスでエラーが発生したときに、ダブルクリックでエラー発生場所に移動できたので何か特殊なデバッグ機能がD3DPreprocessに備わっているのではと考えましたが、単にデバッグ出力を特定の書式にすることで有効になる機能でした。スクリプトやテキストデータの解析時に使えそうです 【受注生産】【アドミレイション/Admiration】 アルファードハイブリッド 等にお勧め アルタモーダ シートカバー LUXURY ラグジュアリー・ストライプ (SSPU×SSPU) 型式等:ATH10W 品番:CS-T040-F 【タナベ】【 サステック タワーバー 】 SUSTEC STURT TOWER BAR ステップワゴン RF3 K20A 2003年06月~2005年05月。