UWP開發細節記錄:DirectX::XMMATRIX 的坑

来源:http://www.cnblogs.com/xrunning/archive/2016/03/06/5247409.html
-Advertisement-
Play Games

這兩天寫的代碼概率性的崩潰在 XMMatrixMultiply() 函數,XMMatrixMultiply() 本身是 inline 函數可以看到崩潰處的代碼: vX = _mm_mul_ps(vX,M2.r[0]); 經查,_mm_mul_ps 是 SSE2 指令要求記憶體地址 16 位元組對齊。猜想


這兩天寫的代碼概率性的崩潰在 XMMatrixMultiply() 函數,XMMatrixMultiply() 本身是 inline 函數可以看到崩潰處的代碼:

vX = _mm_mul_ps(vX,M2.r[0]);

經查,_mm_mul_ps 是 SSE2 指令要求記憶體地址 16 位元組對齊。猜想可能是位元組未對齊引發的問題,列印出矩陣參數的地址,果然,發生崩潰時都是未對齊到 16 位元組的情況。

XMMATRIX 聲明中指定了對齊到 16 位元組,如下:

__declspec(align(16)) struct XMMATRIX 
{
    ...
};

但是顯然沒起到效果,繼續查發現是用到 XMMATRIX 變數被定義成了類成員變數,該類的實例是 new 出來的:

class MyClass
{
   XMMATRIX matrix_; 
};

MyClass* ptr= new MyClass(); // 這裡應該有一個編譯警告

而 new 操作符只保證對齊到不大於 alignof(std::max_align_t)  也就是 8 位元組,導致成員變數 matrix_ 也只能對齊到 8 位元組。 解決方法先將成員變數 martix_ 賦值給一個局部變數,然後通過局部變數調用 XMMatrixMultiply() 函數。直接使用 new 會有一條對象可能無法對齊到 16 位元組的警告信息,但因為我是用 std::make_shared 直接生成智能指針,編譯器沒能給出這條警告——這也是個坑,最終導致了前面的崩潰問題。

註意 XMMatrixMultiply() 的聲明,第一個參數是值傳參,第二個是引用傳參,問題就處在引用傳參上:

1 typedef const XMMATRIX FXMMATRIX
2 typedef const XMMATRIX& CXMMATRIX;
3 
4 inline XMMATRIX XM_CALLCONV XMMatrixMultiply (FXMMATRIX M1, CXMMATRIX M2); //註意第二個參數是引用傳參

MSDN 中對 XMMatrixMultiply 函數的聲明和實際代碼並不一致:

XMMATRIX XMMatrixMultiply(
  [in] XMMATRIX M1,
  [in] XMMATRIX M2
);

如果真的兩個參數都是值傳參也就沒有位元組對齊的問題了。

最終結論:XMMATRIX 結構體要求 16 位元組對齊,只能用作局部變數或全局變數——在靜態存儲區或棧上分配記憶體可以確保 align(16) 聲明有效,永遠不要用 new 分配堆記憶體,也不要用作類成員變數;如果需要在成員變數中保存矩陣可以用 DirectX::XMFLOAT4X4 ,使用時再轉換為 XMMATRIX 。

 

順便吐槽一下,MSDN 以前被稱為寫得最好的開發文檔是當之無愧的,現在只能呵呵了。最近兩年新增的文檔(比如 UWP 和 DX11 的文檔)明顯敷衍了事,函數說明經常一句話了事,看文檔和直接對著函數聲明猜用法都沒什麼區別了,這次居然還出現了函數聲明不一致的情況,這樣下去吃棗藥丸哪。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前兩天弄了個自動配置JDK環境變數的小工具(詳情:http://www.cnblogs.com/chr-wonder/p/5208541.html)。在調試過程中發現了一些小問題。在此做以記錄。 在寫入Path環境變數過程中,由於我是在Path變數的末尾添加新項的,所以為了避免和原有的項衝突(或者說
  • 1、介面: 介面與抽象類一樣,也是表示某種規則,一旦使用了該規則,就必須實現相關的方法。對於C#語言而言,由於只能繼承自一個父類,因此若有多個規則需要實現,則使用介面是個比較好的做法。 2、介面的定義 interface 介面名 { 方法聲明; } 3、不同介面中若有多個相同名稱的方法,則需要顯式指
  • 我想大家對DateTime.ToString()方法的用法肯定已經非常熟悉了,但我想大家用過的大部分用法都是:DateTime.ToString(“format”),不過本文想講述的是它的另一個重載方法DateTime.ToString("format",IFormatProvider)。 如果大家
  • 文本框賦值時,設置其格式非常方便,將值的格式設定好即可。例如: Dim i as integer = 4333 me.textbox1.text = String.Format("{0:00.0}", i) 但對於文本框綁定到數據源時,此方式無效。可以採取下列方法: 在Form的Load事件中,或其
  • 最近公司要做一個項目,需要和現有的其他項目對接,由於不知道他們的資料庫,只有XSD文件。所以,我們在修改相應的程式時,就需要根據他們提供的XSD文件,來寫我們的VO實體類,由於我寫過根據Oracle資料庫生成VO實體類,因此這次的這個活也就很自然的落在了我的頭上。 一、XSD 首先什麼是XSD,我就
  • 摘要: 2016年,Wijmo將專註於:根據客戶的需求打磨控制項,包括性能、功能和修複;增加一些主要的新控制項,比如OLAP、報表、MultiRow;支持Angular 2, Aurelia, EmberJS框架
  • 開啟新的讀書之旅,這次讀的書為《.Net之美:.Net關鍵技術深入解析》。 我是選擇性閱讀的,把一些自己覺得容易忘記的,或者比較重要的知識點記錄下來,以便以後能方便呢查閱。 尊重書本原作者,如果大家能有個可能的話,去看看這本書,作者寫得挺不錯的。例子和知識點各方面都寫挺不錯的。 本章的內容 什麼是委
  • 這篇文章的目的主要為了後續工作的使用,在這裡對S2深入.NET平臺和C#編程中貫穿案例《網路電視精靈》的部分實現代碼進行思路剖析。 一.項目框架的搭建: 這個界面需要的控制項MenuStrip、TreeView、TabControl、ListView、DataGridView。 思路分析: 1.創建對
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...