Btw the endian aware types are usable, just include "llvm/Support/Endian.h".  It's just the consumeObject functions that are not<br><div class="gmail_quote"><div dir="ltr">On Tue, Aug 16, 2016 at 10:01 AM Zachary Turner <<a href="mailto:zturner@google.com">zturner@google.com</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">zturner added a comment.<br>
<br>
LLVM does have something similar to DataExtractor, but unfortunately it's not present in the correct library for you to easily be able to reuse it.  I actually own the code in question, so I'm willing to fix that for you if you're interested, but at the same time the code is very simple.  LLVM has some endian-aware data types that are very convenient to work with and mostly eliminate the need to worry about endianness while parsing, which is what DataExtractor mostly does for you.  To use LLVM's types, for example, you would change this:<br>
<br>
  struct MinidumpHeader<br>
  {<br>
      uint32_t signature;<br>
      uint32_t version;<br>
      uint32_t streams_count;<br>
      RVA stream_directory_rva; // offset of the stream directory<br>
      uint32_t checksum;<br>
      uint32_t time_date_stamp; // time_t format<br>
      uint64_t flags;<br>
<br>
      static bool<br>
      SignatureMatchAndSetByteOrder(DataExtractor &data, lldb::offset_t *offset);<br>
<br>
      static llvm::Optional<MinidumpHeader><br>
      Parse(const DataExtractor &data, lldb::offset_t *offset);<br>
  };<br>
<br>
to this:<br>
<br>
  struct MinidumpHeader<br>
  {<br>
      support::ulittle32_t signature;<br>
      support::ulittle32_t version;<br>
      support::ulittle32_t streams_count;<br>
      support::ulittle32_t stream_directory_rva; // offset of the stream directory<br>
      support::ulittle32_t checksum;<br>
      support::ulittle32_t time_date_stamp; // time_t format<br>
      support::ulittle64_t flags;<br>
  };<br>
<br>
All you have to do now is `reinterpret_cast` the buffer to a `MiniDumpHeader*` and you're good to go.  So pretty much the entirety of the `DataExtractor` class boils down to a single template function:<br>
<br>
  template<typename T><br>
  Error consumeObject(ArrayRef<uint8_t> &Buffer, const T *&Object) {<br>
    if (Buffer.size() < sizeof(T))<br>
      return make_error<StringError>("Insufficient buffer!");<br>
    Object = reinterpret_cast<const T*>(Buffer.data());<br>
    Buffer = Buffer.drop_front(sizeof(T));<br>
    return Error::success();<br>
  }<br>
<br>
For starters, this is nice because it means you're not copying memory around unnecessarily.  You're just pointing to the memory that's already there.  With DataExtractor you are always copying bytes around.  Dump files can be large (even minidumps!) and copying all this memory around is inefficient.<br>
<br>
It also makes the syntax cleaner.  You have to call different functions on DataExtractor depending on what you want to extract.  `GetU8` or `GetU16` for example.  This one function works with almost everything.  A few simple template specializations and overloads can make it even more powerful.  For example:<br>
<br>
  Error consumeObject(ArrayRef<uint8_t> &Buffer, StringRef &ZeroString) {<br>
     ZeroString = StringRef(reinterpret_cast<const char *>(Buffer.front()));<br>
     Buffer = Buffer.drop_front(ZeroString.size() + 1);<br>
     return Error::success();<br>
  }<br>
<br>
I have some more comments on the CL, but I have to run to a meeting, so I will be back later.<br>
<br>
<br>
<a href="https://reviews.llvm.org/D23545" rel="noreferrer" target="_blank">https://reviews.llvm.org/D23545</a><br>
<br>
<br>
<br>
</blockquote></div>