1

64 ビット Delphi XE4 プロジェクト (デバッグ モード) にマップ ファイルを追加する場合。「_zn6」、「_zn11」などの記号をいくつか取得しました。その理由は何ですか? 32 ビット プロジェクトでは、すべて問題ありません。リリース モードを選択した場合、情報は明確ですが、非常に貧弱です。マップ ファイルの一部:

 0005:0000B970       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE6RemoveES5_
 0005:0000B97C       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE10RemoveItemES5_NS_5Types10TDirectionE
 0005:0000B988       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE6DeleteEi
 0005:0000B994       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE11DeleteRangeEii
 0005:0000B9AC       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE7ExtractES5_
 0005:0000B9B8       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE11ExtractItemES5_NS_5Types10TDirectionE
 0005:0000B9C4       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE8ExchangeEii
 0005:0000B9D0       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE4MoveEii
 0005:0000B9DC       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE5FirstEv
 0005:0000B9E8       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE4LastEv
 0005:0000B9F4       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE5ClearEv
 0005:0000BA00       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE6ExpandEv
 0005:0000BA0C       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE8ContainsES5_
 0005:0000BA18       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE7IndexOfES5_
 0005:0000BA24       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE11IndexOfItemES5_NS_5Types10TDirectionE
 0005:0000BA30       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE11LastIndexOfES5_
 0005:0000BA3C       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE7ReverseEv
 0005:0000BA48       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE4SortEv
 0005:0000BA54       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE4SortENS_15DelphiInterfaceINS0_8Defaults12IComparer__1IS5_EEEE
 0005:0000BA60       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE12BinarySearchES5_Ri
 0005:0000BA6C       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE12BinarySearchES5_RiNS_15DelphiInterfaceINS0_8Defaults12IComparer__1IS5_EEEE
 0005:0000BA78       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE10TrimExcessEv
 0005:0000BA84       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE7ToArrayEv
 0005:0000BA90       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE13GetEnumeratorEv
 0005:0000BA9C       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE11TEnumeratorIE10GetCurrentEv
 0005:0000BAA8       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE11TEnumeratorIE12DoGetCurrentEv
 0005:0000BAB4       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE11TEnumeratorIE10DoMoveNextEv
 0005:0000BAC0       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE11TEnumeratorIEC3EPNS2_IS5_EE
 0005:0000BAD8       System.Rtti.$pdata$_ZN6System8Generics11Collections8TList__1IPNS_7Typinfo9TTypeInfoEE11TEnumeratorIE8MoveNextEv

JCL ログのサンプル (x64 マップ ファイルを使用) は次のとおりです。

ERR (ThreadID=12E0 25.01.2014 23:06:28:098) - External exception E06D7363
Exception class: EExternalException
Exception address: 000007FEFD7DBCCD
Stack list, generated 25.01.2014 23:06:27
[000007FEFD7DBCCD] RaiseException + $3D
[00000000775797A8] RtlRaiseException + $248
[000007FEFD7DBCCD] RaiseException + $3D
[000007FEEC70E92C] _CxxThrowException + $D4
[000007FEECB88383] Unknown function at ?GetTotal@ColumnDesc@TablesManager@ProviderEngine@@QEBA_KXZ + $1BF3
[000007FEECB7EB49] Unknown function at DllCanUnloadNow + $33F69
[000007FEECB7B160] Unknown function at DllCanUnloadNow + $30580
[000007FEECB7CC9D] Unknown function at DllCanUnloadNow + $320BD
[00000000030F115D] ConnPool.GetSKRowset (Line 1220, "ConnPool.pas" + 25) + $27
[00000000030F20B2] ConnPool._ZN8Connpool11TConnection9GetResultEN6System15DelphiInterfaceIN6Winapi6Adoint10_RecordsetEEEN11Definitions9TDCReturnENS1_3SetINS_16TRecordsetOptionELSA_0ELSA_2EEEPNS1_7TObjectEb (Line 1341, "ConnPool.pas" + 27) + $1B
[00000000030EEFFF] ConnPool._ZN8Connpool11TConnection18InternalExecuteCmdEv (Line 974, "ConnPool.pas" + 134) + $0
[00000000030EE56E] ConnPool._ZN8Connpool11TConnection7ExecuteEPNS_12TCmdExecArgsE (Line 801, "ConnPool.pas" + 8) + $0
[00000000031C1B97] SKDS._ZN4Skds13TConfigReader7ExecCmdEiN6System13UnicodeStringES2_RKNS1_10OleVariantEPN8Connpool15TCmdExecOptionsEiNS6_9TTranModeEPS3_SA_i (Line 439, "SKDS.pas" + 65) + $0
[00000000031AD053] Dataservice.DoSimpleCall (Line 3241, "Dataservice.pas" + 8) + $167
[00000000031AD929] Dataservice._ZN11Dataservice12TDataservice10RunCommandEiN6System10WideStringES2_RKNS1_10OleVariantES5_RS3_ (Line 3336, "Dataservice.pas" + 44) + $0
[000007FEFE2216D0] Unknown function at SetErrorInfo + $80
[000007FEFE2224D2] DispCallFunc + $262
[000007FEFE221DE1] Unknown function at SetErrorInfo + $791
[0000000002F18242] System.Win.ComObj._ZN6System3Win6Comobj11TAutoObject6InvokeEiRK5_GUIDitPvS6_S6_S6_ + $82
[000000000073407F] Invoker._ZN7Invoker9TKInvoker6InvokeEv (Line 177, "Invoker.pas" + 30) + $73
[00000000007613A5] WorkerThread._ZN12Workerthread14TKWorkerThread17IntCallFromMemoryEN6System15DelphiInterfaceI7IStreamEEii (Line 426, "WorkerThread.pas" + 16) + $0
[0000000000760728] WorkerThread._ZN12Workerthread14TKWorkerThread10WorkInvokeEN6System15DelphiInterfaceI7IStreamEES4_ (Line 391, "WorkerThread.pas" + 59) + $0
[000000000075EEC1] WorkerThread.ProcessRequest (Line 195, "WorkerThread.pas" + 37) + $50
[000000000075F36E] WorkerThread._ZN12Workerthread14TKWorkerThread11DoSomethingEv (Line 218, "WorkerThread.pas" + 4) + $8
[0000000000737038] PoolableThread._ZN14Poolablethread16TKPoolableThread7ExecuteEv (Line 259, "PoolableThread.pas" + 17) + $E
[000000000052C89B] System.Classes._ZN6System7Classes10ThreadProcEPNS0_7TThreadE + $3B
[000000000040DACB] System._ZN6System13ThreadWrapperEPv + $3B
[000000007735652D] BaseThreadInitThunk + $D
[000000007758C521] RtlUserThreadStart + $21
4

1 に答える 1

3

これらの追加の名前は、x64 のさまざまな例外処理モデルに関連しています。x86 では、例外はスタック ベースです。x64 ではテーブルベースです。これは、コンパイラが処理exceptおよびfinallyブロックする方法に影響を与えます。

特に、コンパイラ/リンカーは、例外処理コードを記述する例外テーブルを出力できなければなりません。私が理解しているように、表示される名前は、コンパイラが処理およびブロックするときにコンパイラによって作成さ$pdata$れます。これらの名前は、リンカーによって使用され、実行可能な出力ファイルに書き込まれる例外テーブルが作成されます。そして、コンパイラは、真の関数名と衝突しないように、そのような言葉にできない名前を生成します。$unwind$exceptfinally

私の推測では、スタック トレースにこれらの名前が表示されているのは、JCL スタック ウォーカー コードがこれらの名前を解読できるほど賢くないためだと思います。たとえば、madExcept を使用していた場合、期待していた名前が表示されます。

根本的な問題は、JCL に機能がないことです。


x86 と x64 の構造化例外処理には大きな違いがあります。たとえば、finallyブロックのコンパイル済みコードが x64 実行可能ファイルに 2 回出現することは興味深い事実です。この短いプログラムを考えてみましょう:

procedure Foo;
begin
end;

procedure Main;
begin
  try
  finally
    Foo;
  end;
end;

begin
  Main;
end.

コンパイラは次のように変換Mainします。

Project1.dpr.8: 開始
0000000000409A30 55プッシュrbp
0000000000409A31 4883EC30 サブルプ、$30
0000000000409A35 488BEC mov rbp,rsp
0000000000409A38 48896D28 mov [rbp+$28],rbp
Project1.dpr.9: 試してみる
0000000000409A3C 90 nop
Project1.dpr.11: フー;
0000000000409A3D 90 nop
0000000000409A3E E8DDFFFFFF 呼び出し Foo
Project1.dpr.13: 終わり;
0000000000409A43 488D6530 リー rsp,[rbp+$30]
0000000000409A47 5D ポップ rbp
0000000000409A48 C3 ret
0000000000409A49 488D8000000000 リーラックス,[rax+$00000000]
Project1.dpr.11: フー;
0000000000409A50 55プッシュrbp
0000000000409A51 4883EC20 サブルプ、$20
0000000000409A55 488BEC mov rbp,rsp
0000000000409A58 E8C3FFFFFF 呼び出し Foo
0000000000409A5D 488D6520 lea rsp,[rbp+$20]
0000000000409A61 5D ポップ rbp
0000000000409A62 C3 ret

への 2 つの呼び出しに注意してくださいFoo。最初のものは通常の実行です。これは、例外がなく、finally通常どおりブロックに入る場合です。への 2 番目の呼び出しFooは、例外がアクティブな場合を処理します。

この finally ブロックの 2 番目のバージョンは、実際には別の関数としてコンパイルされます。Project1.$pdata$_ZN8Project13FooEv私のマップファイルによると名前が付いています。

0005:00000A50 Project1.$pdata$_ZN8Project13FooEv

これは、メインの例外ハンドラから呼び出されSystem._DelphiExceptionHandlerます。そして、それが で終わっているという事実からわかるように、それは実際には別の機能retです。このバリアント コードを実行するために内部で例外をスローするtry/finallyと、IDE のスタック トレースは次のようになります。

Project1.Main
System._DelphiExceptionHandler($12FAB0,1244912,$12E820,$12E730)
:00000000779F9DAD; ntdll.dll
:00000000779E8A4C; ntdll.dll
:00000000778E2D3E; C:\Windows\system32\kernel32.dll
System._DelphiExceptionHandler($12FAB0,1244976,$12F5C0,$12EF70)
:00000000779F9D2D; ntdll.dll
:00000000779E91CF; ntdll.dll
:0000000077A21248; ntdll.dll
:000007FEFDA7940D; C:\Windows\system32\KERNELBASE.dll
System._RaiseAtExcept(???,???)
System._RaiseExcept(???)
Project1.Main

ご覧のとおり、IDE は JCL コードでは不可能なことを実現し、テーブル ベースの例外処理を理解することができます。

x86 ではかなり異なって見えます:

Project1.Main
Project1.Project1
:7618336a kernel32.BaseThreadInitThunk + 0x12
:77be9f72 ntdll.RtlInitializeExceptionChain + 0x63
:77be9f45 ntdll.RtlInitializeExceptionChain + 0x36

したがって、これらの言いようのない名前はすべて、テーブル ベースの例外の管理に関連しています。この動作は完全に予期されたものです。

于 2014-01-26T11:30:06.550 に答える