簡體   English   中英

為什么使用帶有以下LLVM庫代碼的帶有-O3 segfault的g ++

[英]Why g++ with -O3 segfault with the following LLVM library code

所以我想使用llvm::Twine字符串塊類。

我有以下示例:

#include <llvm/ADT/Twine.h>
#include <iostream>

int main()
{
  llvm::Twine twine1 = llvm::Twine("aaaa") + "bbbb" + "cccc" + "dddd";
  llvm::Twine twine2 = llvm::Twine(twine1) + "dddd" + "eeee";
  std::cout << twine1.str() << std::endl;
  std::cout << twine2.str() << std::endl;

  return 0;
}

它運行與clang++-O3g++-O0但與段錯誤g++-O3 我嘗試了這段代碼部分的3.4-3.9版本的clang庫,並嘗試了g++ 4.8.4g++ 4.8.5mingw-5.3.0

您需要llvm庫,並將代碼與-lLLVMSupport -lLLVMCore以及其他來自llvm-config --ldflags

Twine文檔中

Twine不能直接使用,也不應該存儲,它的實現依賴於存儲指向臨時堆棧對象的指針的能力,該指針可以在語句的末尾釋放。 僅當API希望接受可能串聯的字符串時,才應將Twines用作參數中的const引用。

換句話說,Twine對象不擁有其各個部分,因此它們在語句末尾被銷毀。

正確的用法是:

#include <llvm/ADT/Twine.h>
#include <iostream>

void bar(const llvm::Twine& twine1, const llvm::Twine2& twine2){
    std::cout << twine1.str() << std::endl;
    std::cout << twine2.str() << std::endl;
}

void foo(const llvm::Twine& twine1){
    bar(twine1, twine1 + "dddd" + "eeee");
}

int main()
{
  foo(llvm::Twine("aaaa") + "bbbb" + "cccc" + "dddd");

  return 0;
}  

看起來您正在嘗試使用gcc編譯一些代碼,並將其鏈接到使用其他編譯器構建的庫。 llvm

不同的編譯器通常會使用不同的內部ABI 通常,除非有明確的ABI兼容性保證,否則一個C ++編譯器編譯的代碼不能與其他編譯器構建的代碼鏈接。 TMK,在gcc和llvm之間沒有。 即使實際鏈接可能成功,結果也不會具有定義的行為。

實際上,由一個版本的gcc構建的代碼通常無法與由另一個版本的gcc構建的代碼鏈接(許多版本的gcc確實具有某些ABI兼容性保證,但不能與所有其他版本的gcc兼容)。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM