[LLVMdev] llvm-mc direction

Chris Lattner clattner at apple.com
Mon Jul 6 18:51:46 PDT 2009


Hi All,

Bruno asked me to write up a short email about the direction we're  
going with llvm-mc and how it interacts with the object writer stuff.   
My hope was that .o writing could be basically done without worrying  
about llvm-mc, but I think we've reached the point where it is useful  
to talk about how the two will interact in the future.  It turns out  
that I also failed in the goal of writing a *short* email, oh well.

llvm-mc has many goals, but it basically revolves around generating  
full native assemblers and disassemblers from suitably enhanced  
versions of the .td files we already use for the code generator.  For  
this email, I'll just focus on the native assembler part.

If you look at writing out a .o file, there are a couple of  
interesting parts: 1) encoding machine instructions, 2) wrapping an  
ELF/PECOFF/MACHO container around them, 3) supporting inline assembly,  
and 4) wiring it into the LLVM backend.

LLVM already handles #1 fairly well.  It can certainly be improved,  
but the JIT works today.

Aaron and Bruno are working on #2, integrating .o file writing into  
the backend but ignoring inline assembly.  The integration into llvm  
(#4) is pretty ugly, because the .o file writers duplicate a ton of  
logic in the asmprinters.  I intentionally asked Bruno to ignore  
inline assembly.

My goal with llvm-mc is to solve all of these problems, pulling in  
Bruno and Aaron's work on #2 when the time is right. The ultimate  
design I'd like to get to looks like this:

1. Implement a full assembler, with a .s parser, a fragment processor,  
and .o file writer.  The main data-structure the .o file writer will  
work on are "fragments" which (at the end of the assembler) are  
basically chunks of bytes with an associated section and a list of  
relocations (which is exactly what llvm::BinaryObject is!).

2. The full assembler should work as a drop in replacement for  
"as"/"gas".  This allows easy testing, and may also be interesting to  
the BSD folks.  I am slightly worried and paranoid about getting  
different output when using "llvm-gcc -S + as" and using "llvm-gcc - 
c", so testing is very important to me.

3. The assembler needs an API between the .s parser and the "assembler  
backend".  This is called MCStreamer so far - it is somewhat similar  
to the "Actions" interface in clang.  This separates the "parsing"  
from the "fragment construction", relaxation, and other stuff that an  
assembler does.

4. Since we have a nice MCStreamer API, we plan to implement it in two  
ways: one way will use the assembler backend to write out a .o file.   
The other way will just *prints out a .s file* that is semantically  
identical to the input.  This is currently starting to limp along, and  
is a good way to bring up the asm parser without worrying about the  
"assembler backend" and .o file writing.

5. The LLVM "asmprinter" will change from writing out text directly  
with raw_ostream calls to making virtual method calls on the  
MCStreamer API.  For example, instead of 'O << "\t.align 2\n";', the  
asmprinter will end up calling Streamer->EmitValueToAlignment(2).   
When run in "-S" mode, the implementation of the Streamer will be  
the .s file writer, which just writes out the .align directive.

6. When the "assembler backend" comes up, this means that we can drop  
it into the current "asmprinter" and the virtual methods (like  
EmitValueToAlignment) will just do the right thing for building a .o  
file.  This also means that the compiler can pull in the asmparser to  
handle inline assembly with no problem.

Taking this sort of approach has a number of advantages.  A big  
problem that I see today is that the .s writer and .o writers use  
completely different code paths that make almost exactly the same  
decisions.  For example, if separate, they both need to have some  
equivalent of the X86ATTAsmPrinter::printModuleLevelGV function, which  
decides what low-level linkage, visibility, and many other aspects of  
a global to emit.

Duplicating all this logic is bad and is likely to lead to -c and -S  
doing different things.  If the only difference between -c and -S is a  
different implementation of MCStreamer, and if we can independently  
test that the assembler "does the right thing", we should be in good  
shape for correctness.

There is clearly some potential for overlap between what the MC stuff  
is doing and the .o file writing work is doing.  Before I talk about  
that, I want to talk about the implementation plan for MC.  The plan  
is to do X86-32 first then x86-64, then move on to ARM probably, with  
these steps (many of which are parallel):

1. Make a new llvm-mc test tool that is a driver for various  
components (done).
2. Build up an asm parser for x86 that can print out a semantically- 
identical .s file (one that can be assembled to the same bytes as the  
input).  This "complicated cat" is just a testing mechanism, but it  
also allows implementation of the asm parser to proceed in parallel  
with other pieces, and will be used by "-S" eventually (in progress).
3. Implement support for tblgen-generating the parsing logic that maps  
from "opcode name + argument" in the asmparser to an instruction enum  
value + MCOperands.
4. Refactor the asm printer to go through MCStreamer instead of  
writing to raw_ostream directly (in progress).
5. Refactor the asm printer to not depend at all on libx86, codegen,  
vmcore etc.  The goal is for x86/asmprinter to only depend (at a link  
level) on libsupport.  There is a lot involved in this, to say the  
least, but this is important for "llvm-mc as a disassembler API".
6. Implement the "assembler middle end", which handles relaxation  
operations (e.g. determining whether something is a short or long  
branch on x86, figuring out how big a uleb has to be, etc), which  
generates the final fragments (which are basically going to be  
llvm::BinaryObject's).
7. Implement support for writing a .o file.

As you can see, llvm-mc is a pretty ambitious and non-trivial  
project.  Because of this, I don't expect to get to #7 for a couple  
months at least, and I understand that the .o file work needs to be  
almost complete by then (for GSoC etc).  My goal was for the .o  
writing work to focus on stuff that doesn't intersect with MC stuff  
until late in the game: the encoding to ELF, writing DWARF as bytes  
instead of ".byte's" etc.  My hope was that the .o file writer stuff  
could just be refactored directly into providing #7, completing the  
final step.

Coming back to the short-term issue of "we need templates to make byte  
emission fast in the code generator", I hope that I've convinced you  
that this will "just not matter".  Please focus on making .o file  
writing be done in a simple, clear and concise way without worry about  
micro-level performance optimizations.  By the time we're done with  
this whole project, many things will have changed.  Doing optimization  
of .o file writing at this stage is *way way way* too early.

One nice thing about this design is that (like clang) it will be very  
easy to decompose the layers and performance test just (e.g.) the .o  
writer without testing the whole compiler's performance.

-Chris



More information about the llvm-dev mailing list