LLVM。如何根据结构字段的名称访问它们?

LLVM。如何根据结构字段的名称访问它们?,llvm,llvm-c++-api,Llvm,Llvm C++ Api,我有一些C++的示例代码: struct RecordTest { int value1; int value2; }; void test() { RecordTest rt; rt.value1 = 15; rt.value2 = 75; } 以及针对it的LLVM 3.4 IR: %struct.RecordTest = type { i32, i32 } ; Function Attrs: nounwind define void @_Z4t

我有一些C++的示例代码:

struct RecordTest
{
    int value1;
    int value2;
};

void test()
{
    RecordTest rt;
    rt.value1 = 15;
    rt.value2 = 75;
}
以及针对it的LLVM 3.4 IR:

%struct.RecordTest = type { i32, i32 }

; Function Attrs: nounwind
define void @_Z4testv() #0 {
entry:
  %rt = alloca %struct.RecordTest, align 4
  %value1 = getelementptr inbounds %struct.RecordTest* %rt, i32 0, i32 0
  store i32 15, i32* %value1, align 4
  %value2 = getelementptr inbounds %struct.RecordTest* %rt, i32 0, i32 1
  store i32 75, i32* %value2, align 4
  ret void
}
还有一个非常简单的问题:如何访问
RecordTest
字段(当我解析.cpp时),而不使用它们的索引,只使用名称(
value1
value2

我只知道一种方法(来自
llc-march=cpp
)-索引:

  AllocaInst* ptr_rt = new AllocaInst(StructTy_struct_RecordTest, "rt", label_entry);
  ptr_rt->setAlignment(4);
  std::vector<Value*> ptr_value1_indices;
  ptr_value1_indices.push_back(const_int32_6);
  ptr_value1_indices.push_back(const_int32_6);
  Instruction* ptr_value1 = GetElementPtrInst::Create(ptr_rt, ptr_value1_indices, "value1", label_entry);
  StoreInst* void_9 = new StoreInst(const_int32_7, ptr_value1, false, label_entry);
  void_9->setAlignment(4);
  std::vector<Value*> ptr_value2_indices;
  ptr_value2_indices.push_back(const_int32_6);
  ptr_value2_indices.push_back(const_int32_5);
  Instruction* ptr_value2 = GetElementPtrInst::Create(ptr_rt, ptr_value2_indices, "value2", label_entry);
  StoreInst* void_10 = new StoreInst(const_int32_8, ptr_value2, false, label_entry);
  void_10->setAlignment(4);

是这样吗?

不能按名称访问结构的字段,只能按索引访问。当您使用Clang编译时,这些信息通常不存在

这有一个例外,如果使用调试信息编译。在这种情况下,您将拥有关于该类型的大量数据;具体地说,您将获得字段的顺序,以及每个字段的元数据条目,其中包含其名称(以及其他有用的内容,例如从类型开头的偏移量)

在指南中阅读更多关于这方面的内容,特别是,请参见,其非常好的示例


查看一下有哪些类可以帮助您查询调试信息,不过我认为您还是需要手动进行挖掘。

谢谢!那么,在这种情况下,我应该如何在没有调试信息的情况下访问这些字段呢?我更新了一个问题,如果你有一点空闲时间,请看一看。@kpdev我不明白,你首先将如何填充
mymap
?我的意思是,在没有调试信息的情况下,从何处获得有关其字段的信息?从解析.cpp文件。(或者其他语言的源文件——例如,我为Oberon风格的语言编写了一个LLVM编译器)。我们可以在这一步中填充mymap(为每个新结构获取pairs字段索引)@kpdev如果您使用的是Clang,更好的解决方案是使用调试符号进行编译。如果您使用的不是Clang而是您自己的前端,那么是的,正确的方法是在解析过程中对数据进行编码(另请参阅),然后您可以从LLVM中提取数据。是的,我使用的是我自己的前端(实际上是我编写的)。谢谢你的链接,我会立即查看)
 // It can be some kind of singletone
static std::map<std::string, std::vector<std::string>> mymap;

// Some function, where we first time meet RecordTest
    std::vector<std::string> fieldNames;
    fieldNames.push_back("value1");
    fieldNames.push_back("value2");
    mymap["RecordTest"] = fieldNames;

// Some function, where we need to access to RecordTest field
    std::vector<std::string> fieldNamesAgain = mymap.find("RecordTest")->second;
    std::string fieldName = "value1";
    int idxValue1 = -1;
    for (int i = 0, e = fieldNamesAgain.size(); i < e; i++) // little ugly search
    {
        if (fieldName == fieldNamesAgain[i])
        {
                // we get field index, and now we can build code
                // as in example above (llc -march=cpp)
            idxValue1 = i;  
            break;
        }
    }