自己动手写编译器

这里 不再仅仅是 简单的记录一下……
  1. 直接上手
    1. 环境
    2. 目标1:在elicpse平台上使用ant构建ANTLR
    3. 目标2:在elicpse平台上使用antlride编写ANTLR语法
  2. 编写实用的 C解析器
    1. 背景调查
    2. 到底还需不需要自己动手写一个编译器
      1. 这就是“工具哲学”
      2. 所谓bootstrap的编译器构造方式。
      3. C的核心子集
  3. Appendix

直接上手

环境

antlr-for-eclipse,即antlride-2.0-rc4版,在eclipse-modeling软件包中使用还需要dltk包支持
ANTLR v3版,包括一个jar文件(包含runtiime和sdk和gunit,而在2.1。7版时,还是分开的4个jar文件)、一个antlr3.jar的ANT任务库文件(也是来自antlr的主页)
eclipse ,分别在ganymede和galileo两个版本的modeling_incubation和SDK上验证通过
Windows 7 / Windows XP+java6u16和Ubuntu 9.10+sun-java6 上分别验证通过

其它工具的下载链接、安装方式   TODO

目标1:在elicpse平台上使用ant构建ANTLR

  1. 以Windows 版为例:(Ubuntu版在设置ANTLR_HOME时,可以通过输入ANTLR_HOME=xxx ./eclipse方式启动,不需要修改系统环境变量)
  2. 在系统属性中添加ANTLR_HOME环境变量,如antlr-3.2.jar位于F:workspaceESAlib,则设为:F:workspaceESA
  3. 将antlr3.jar这个ant任务文件复制到eclipse的ant库下,如:F:modelingpluginsorg.apache.ant_1.7.1.v20090120-1145lib
  4. 在eclipse的ANTruntime配置中,单击ANT_HOME,重新定位之,使ANT重新扫描,发现新加入的任务antlr3并添加之。这时候会提示错误,再加入JDK6的tools.jar.
  5. 编写build.xml,形如:
  xml version="1.0" ?>
- < project name =" c " default =" generate " basedir =" . " >
  < property name =" version " value =" 1.00 " />
  < property name =" src " location =" src " />
  < property name =" bin " location =" bin " />
  < property name =" lib " location =" lib " />
  < property name =" build " location =" build " />
  < property name =" dist " location =" dist " />
  < property name =" doc " location =" doc " />
  < property name =" grammar " location =" grammar " />
- < target name =" init " >
-
  < tstamp />
  < mkdir dir =" ${bin}/META-INF " />
-
- < fail message =" Environment variable ANTLR_HOME is not set! " >
- < condition >
- < not >
  < isset property =" envList.ANTLR_HOME " />
  not >
  condition >
  fail >
  target >
-
-
-
  < property environment =" envList " />
-
  < property name =" antlrHome " value =" ${envList.ANTLR_HOME} " />
- < patternset id =" antlr.libs " >
  < include name =" antlr-*.jar " />
  < include name =" stringtemplate-*.jar " />
  < include name =" runtime-*.jar " />
  patternset >
- < path id =" antlr.path " >
- < fileset dir =" ${antlrHome}/lib " casesensitive =" yes " >
  < patternset refid =" antlr.libs " />
  fileset >
  path >
-
  < property name =" report " value =" true " />
  < property name =" multithreaded " value =" true " />
  < property name =" debug " value =" false " />
-
- < macrodef name =" antlr3 " >
  < attribute name =" grammar.name " />
  < attribute name =" package.dir " />
- < sequential >
  < echo message =" antlr ${grammar}/@{grammar.name} " />
- < antlr:antlr3 xmlns:antlr =" antlib:org/apache/tools/ant/antlr " target =" ${grammar}/@{grammar.name} " outputdirectory =" ${src}/@{package.dir} " libdirectory =" ${src}/@{package.dir} " multithreaded =" ${multithreaded} " report =" ${report} " debug =" ${debug} " >
- < classpath >
  < path refid =" antlr.path " />
  classpath >
  < jvmarg value =" -Xmx512M " />
  antlr:antlr3 >
  sequential >
  macrodef >
-
-
- < target name =" generate " depends =" init " description =" using ANTLR to generated *.java " >
  < antlr3 package.dir =" per/chenjw/esa/front/c " grammar.name =" C.g " />
  target >
- < target name =" compile " depends =" init, generate " description =" compile *.java generated by ANTLR. " >
- < javac debug =" true " srcdir =" ${src} " destdir =" ${bin} " listfiles =" Yes " deprecation =" Yes " >
  < compilerarg value =" -Xlint:unchecked -nwarn " />
- < classpath >
  < path refid =" antlr.path " />
  classpath >
  javac >
  target >
-
- < target name =" clean " description =" Removes all generated and disttribution files. " >
  < delete dir =" ${bin} " />
- < delete >
- < fileset dir =" ${dist} " >
  < include name =" *.jar " />
  < exclude name =" *.txt " />
  fileset >
  delete >
  target >
  project >
  1. Alt+X+Q运行之。


FAQ

常见错误如报错 ant任务不识别,这主要是没有重新扫描ant的lib目录造成的。

TODO

antlr3通过gunit支持更加便捷的测试途径,更详细的ant脚本在这@gmail.com。

目标2:在elicpse平台上使用antlride编写ANTLR语法

  1. 将antlride和ltdk两个的插件包分别放置到eclipse-modeling平台的dropins里面
  2. 启动eclipse后修改ANTLR的配置,指定ANTLR runtime环境,如F:workspaceESA,并制定antlr的输出文件路径。需要注意的是,这里的路径要跟.g文件中header段指定的包名一致。
  3. 在项目如ESA右键中添加ANTLR IDE support,这时候在buiild project操作时,就会包含.g文件的编译。

 

Suggestion
为了使antlride和ant方式的输出保持一致,不得不将所有的语法文件生成的包放置到同一个路径下,因此,在.g语法文件中声明的package也必须统一。至于tokens,则不是必须。

编写实用的 C解析器

背景调查

网上有人提供的 ANTLR自学经验、资料 ,包含一个tiny-C的语法文件。不过,里面的内容确实很老了,但是一些“老人谈”还是成立的:利用ANTLR编写一个实用的解析器并不比利用Yacc/Lex编写容易太多,作者本人就 因为调试语法不成放弃了 。正如下图所示,编译器技术本来就是一件“屠龙技”。

当然,辅助编写语法的东西还是有的,比如上面网址提供了一个 AST树美化输出程序()


eclipse 上的 JDT中关于Java 解析的原理、实现

 

Flick: The Flexible IDL Compiler Kit

    utah大学提供的IDL 编译器助手库:

Program Database Toolkit

Oregon大学提供的程序数据库工具,架构如下。该工具使用下面EDG公司的解析引擎,它自身的重点在于后端的分析、插桩、组装环节。


Thoughts on the Visual C++ Abstract Syntax Tree (AST)


Parsing C++ 最后,如果看完上面的内容都没有什么收获的话,那么最值得一看就是这篇,作者详细说明了他试图开发一个refactoring C++工具的前因后果,很有深度和概括性。
文中介绍了很多启发性的东西,比如元解析----不是直接打造一个完整的解析器,而是迭代迭构造,思路来自于一篇博士论文。而且这样可以构造出一个比原始C++更大的集合,这可以用来是实现 如嵌入式领域C++语言的扩展。文章提到一个 Edison Design Group公司 ,主页上说的话非常贴近心声,强烈推荐。


到底还需不需要自己动手写一个编译器

看到上面那些激动人心的网络资料或许让人产生伤心沮丧的感觉,----到底还需不需要自己动手写一个编译器?
这个问题或许不能从应用寻找答案,于是向程序员本身寻找答案:

程序员都是实在人,他们不会问“东西作出来到底有没有用”,他们只会小声的问一句:我如何重用之前的语法呢?
实际上,ANTLR v2就提供了 subgrammar 的构造用于grammar reuse。但是,在实际使用中并不顺利,eclipse CDT项目团队的队长 Chris Recoskie 叙说:
The one thing that sort of got in the way of [grammar reuse] in ANTLR 2 was that if you overrode a rule, you had to override the action as well in its entirety.
实现重用,并不容易;
但是,语法重用并不是唯一的选择。 Chris Recoskie 也说出了另一种“重用”的途径:
Ideally I’d like someone to have my C99 grammar sitting in their workspace, and if ever I bug fix my grammar, they just automagically pick up all of those changes when they build their grammar again, including all the semantic actions.”

这种思路的详细介绍在这篇论文中阐述 The Reuse of Grammars with Embedded Semantic Actions 。

这篇论文中一段关于设计哲学的话放在这里思考非常具有启发性:
The philosophy of the ANTLR project is to formalize and automate what programmers do naturally and to present solutions that fit the way programmers think about a partic ular problem. In this case, being able to pick up changes from a grammar stored in a central repository smacks of source code revision control.

这就是“工具哲学”
事实上,所有的理论都有着或近或远、或普遍或专门的应用,而且,所有复杂的理论都存在十分直观、朴实的逻辑。在计算机领域,一个典型的逻辑就是:分而治之,逐步求精
所谓bootstrap的编译器构造方式。
关于如何做bootstrap编译, Bootstrapping a simple compiler from nothing (也可见 附录 )是一个相当生动、具有说服力的例子,完全从加载十六进制码开始bootstrap,十分神奇。类似的,我们先打造一个C语言子集的解析语法,然后再利用这个子解析语法构造能够解析C语言扩展集。具体来说,就是先提取C语言中的基础的、简单的部分出来,包括expression、while、goto、if-else出来,编写一个解析语法器,它识别的语法只是C语言的一个子集;然后,进一步考虑更广泛的C语言语法,如何映射到先前的这个子集上来,如何转换。这个解析器是最终的解析器,必须能够接受更加丰富多样的特化,比如以字符串作为+的操作对象、=>这样的赋值操作符、|既可以作为逻辑运算符也可以作为两个函数并行执行的操作符、在函数定义后面、实现体之前插入interuupt 1 using 3这样C51的构造语法等
这时的工作包括两部分,主要部分是一种转换,比如将for 循环用等价的while语句和赋值等替换;另外一部分更为重要,就是添加,加入新的语法推导路线,在原来的语义上加入新的元素,或者添加一套新的语义元素及相应操作方式。

在ANTLR编程的具体实现上,这种链式传递关系完全可以通过stringTemple的方式实现,即把上一轮解析的结果作为一种新的language保存,然后对它进行解析,如此迭代。如果只需要两轮,那么也可以通过parser grammar + tree grammar的方式实现,相当灵活。

C的核心子集

在C解析实践中,动态部分是比较好做的,而静态部分---- type system (包括declaration、inference、implicit cast)则不太好做。原因就在于,类型系统本身就是形式化方法在程序设计中的最初、也是最成功的应用。要搞定这部分,光靠BNF、LL(*)这些形式化东西是不够的。







 



Appendix

Bootstrapping a simple compiler from nothing

============================================

This document describes how I implemented a tiny compiler for a toy
programming language somewhat reminiscent of C and Forth. The funny
bit is that I implemented the compiler in the language itself without
directly using any previously existing software. So I started by
writing raw machine code in hexadecimal and then, through a series of
bootstrapping steps, gradually made programming easier for myself
while implementing better and better "languages".

The complete source code for all the stages is in a tar archive:
. This text is the README file
from that archive . So, if you are reading this on-line, you can fetch
the tar archive and continue off-line, if you prefer.

The code only runs on i386-linux, though it would be easy to port it
to another operating system on i386, and probably not at all hard to
port it to a different architecture.


HEX1: the boot loader
---------------------

You could input a short program into the memory of an early computer
by using switches on its front panel. This short program might then
read in a longer program from punched cards. To write a program on
punched cards you did not need an editor program, as you could write
new cards using an electro-mechanical card punch and manually insert
and remove cards from the deck. So, if we were using an early
computer, we could really implement a compiler without using any
existing software. Unfortunately, a modern PC has neither front panel
switches nor a punched card reader, so you need some software running
on the machine just to read in a new program. In fact, you probably
need some rather complex software running on the machine: just take a
look at /usr/src/linux/drivers/block/floppy.c, for example.

Since we are doing this on a PC running linux , we have to define some
other starting point. Rather than use the raw hardware, we start with
these facilities:

- an operating system;

- a simple text editor (or we could use Emacs and pretend it's a
simple text editor);

- a shell that lets us run a program with file descriptors connected
to particular files (this way the programs we write only need to
read from and write to file descriptors and do not have to know
about opening files);

- an initial program to convert hexadecimal to binary so that we can
compose our first programs in hexadecimal, using the text editor,
and then "compile" them to binary in order to run them (this
corresponds roughly to the program that you might enter into an
early computer using front panel switches).

Our initial program is hex1.he (the source in hexadecimal) or hex1
(the binary). If you want to check that hex1 really is the binary
corresponding to hex1.he, you can do a hex dump of it:

od -Ax -tx1 hex1

If you use hex1 to process hex1.he the result it hex1 again:

./hex1 < hex1.he | diff - hex1

So we can think of hex1 as a trivial bootstrapping compiler for a
language called HEX1.

Apart from comments and white space, the syntax of HEX1 is
/([0-9a-f]{2})*/. Comments start with '#' and continue to the end of
the line. The semantics of HEX1 is the semantics of machine code,
which is rather complex. Fortunately we can restrict ourselves to a
tiny subset of the full instruction set.

In hex1.he I have put the corresponding assembler code in comments
next to the machine code. The file starts with two ELF headers: a
52-byte file header and a 32-byte program header. It is not necessary
to understand all the fields in the ELF header. The most interesting
fields are:

* e_entry, which specifies where execution should begin. Here it is
0x08048054, which is directly after the ELF headers (labelled _start).

* p_vaddr and p_paddr, which specify the target address in memory.
Here it is 0x08048000, which is standard for linux binaries.

* p_filesz and p_memsz, which should be set to the length of the file.
It seems not to matter if you put a larger number here, and I will
make use of that later, though here I have put the correct value.

(For more information about ELF do a web search. SCO and Intel have
some useful on-line documents.)

The code at _start is a loop that reads pairs of hex digits by calling
gethex and outputs bytes by calling putchar. Next comes putchar, which
uses the "write" system call. Then gethex, which calls getchar and
contains a loop for skipping over comments. The ASCII characters
[0-9a-f] are converted correctly to the values 0 to 15; everything
below '0' (48) is treated as a space and ignored; other characters are
misconverted, as there is no error detection. The function getchar
uses the "read" system call, and calls "exit" at the end of the file.


HEX2: one-character labels
--------------------------

Writing machine code in hex is not much fun. The worst part is
calculating the addresses for branch, jump and call instructions. Here
I am using relative addresses, so I have to recalculate the address
every time I change the length of the code between an instruction and
its target. It would be no better if I were using absolute addresses:
then I would have to change all references to locations after the
change.

So the first feature I add for my convenience is a function for
computing relative addresses. Instead of writing

# function:
...
e8 cc ff ff ff # call function

I will be able to write:

.F # function:
...
e8 F # call function

HEX2 automatically fills in the correct 4-byte relative address.

Unfortunately, I still have to use HEX1 to implement the first version
of HEX2, so, to keep the implementation simple, I only allow
one-character labels and backwards references to them. And there is no
error detection for an undefined label.

The syntax of HEX2 is ([0-9a-f]{2}|.L|L)*, where L is any character
above 32 apart from [0-9a-f].

The first implementation of HEX2 is hex2a.he. If you compare the ELF
headers in hex1.he and hex2a.he you will notice that I have changed
p_flags. This is to make the program writable as well as executable.
Normal programs consist of several sections, in particular a text
section, which contains the program itself, and a data section. The
text section is executable, but not writable, and the data section is
writable, but not executable. In hex1.he I did not need to write any
data to memory, so I only had a text section. In hex2a.he I need to
write data to memory, but I can not be bothered with separate
sections, so I use a single section which is both executable and
writable.

There are only two pieces of data: "pos" is a 32-bit counter to keep
track of our location as we output the binary, and "label" is a
259-byte table to record the values of the labels. Why 259 bytes? This
is because I forgot to multiply by 4. I should have used a table of
256 4-byte values, one for each possible one-character label, and
calculated the address as (table + char * 4). Since I forgot to
multiply by 4, I only need 259 bytes for my table, and I have to avoid
using labels that are close to one another: if I use 'm', then I
cannot use 'j', 'k', 'l', 'n', 'o' or 'p'. It would be easy to fix
this bug immediately, but it is even easier to work around it for now
and fix it a bit later.

We can "compile" hex2a.he using hex1:

./hex1 < hex2a.he > hex2a && chmod +x hex2a

Since HEX2 is a superset of HEX1, hex2a.he can also compile itself:

./hex2a < hex2a.he | diff - hex2a

To test the new facility, I made hex2b.he from hex2a.he by replacing
numerical addresses by symbolic ones wherever possible. Compiling
hex2b.he gives the same binary as hex2a.he:

./hex2a < hex2b.he | diff - hex2a

In hex2c.he I fix the "multiply by 4" bug. It is easier to fix the bug
now that I can use labels and do not have to manually modify relative
addresses. In hex2c.he I also replace some 1-byte relative addresses
by 4-byte relative addresses, so that I can use labels, and I have
inserted blocks of NOPs at the end of file to make the precise value
of e_entry less critical.

We can compile hex2c.he using hex2a/hex2b or using itself:

./hex2a < hex2c.he > hex2c && chmod +x hex2c
./hex2c < hex2c.he | diff - hex2c


HEX3: four-character labels and a lot of calls
----------------------------------------------

One-character labels are a bit restrictive, so let us implement
four-character labels. If labels have exactly four characters we can
store them neatly in 32-bit words!

The syntax of HEX3 is /([0-9a-f]{2}|:....|.....)*/, and now we will
introduce some very basic error detection. The compiler can report
three different kins of error, which is will do using its exit code:

exit code 1: syntax error
exit code 2: redefined label
exit code 3: undefined label

Since it is a single-pass compiler, only backwards references to
labels are permitted.

The first implementation of HEX3 was hex3a.he, written in HEX2:

./hex2c < hex3a.he > hex3a && chmod +x hex3a

It is not possible to compile hex3a.he with hex3a itself, as HEX3 is
not compatible with HEX2.

I created hex3a.he by making successive small changes to hex2c.he. The
system call brk() is used to get memory for an arbitrarily large
symbol table. Absolute references to data are avoided by putting a
function (.z / get_p) in front of the static data area that returns
the address of the following data.

Having created hex3a.he, I started work on hex3b.he, an implementation
of HEX3 written in HEX3. Initially hex3b.he was just hex3a.he
translated to the new syntax, but I then gradually rewrote it to make
much greater use of labels and functions. In the final version, after
a certain point in the file, everything is done using only these
instruction groups:

- push a constant onto the stack: 68 XX XX XX XX
- call a named function: e8 .LABEL
- unconditional jump: e9 .LABEL
- conditional branch: 58 85 c0 0f 85 .LABEL
- push an address onto the stack: 68 .LABEL e8 .reab

The last instruction group consists of a push instruction followed by
a call instruction, but the two may not be separated: the function
"reab" converts the relative address on the stack to an absolute
address by adding its return address and subtracting 5.

We can compile hex3b.he using hex3a or itself:

./hex3a < hex3b.he > hex3b && chmod +x hex3b
./hex3b < hex3b.he | diff - hex3b


HEX4: any-length labels and implicit calls
------------------------------------------

When implementing hex3b.he we found that it is possible to define all
complex functions in terms of simpler functions by using a tiny subset
of all the possible machine instructions: branch, call, jump and a few
others.

In HEX4 we use an even smaller set of instructions and generate those
instructions implicitly.

In HEX4 there are four types of token:

- in-line code or data ('58, '59)
- define label (:data, :loop, :func)
- instruction: push constant (10, 42)
- instruction: push label address (&func, &loop)
- instruction: call label address (+, -, jump, branch, func)

Tokens must be separated by white space and the type of token is
recognised from the first character. Labels can have any length - but
we implement them with a simple hash function, so there is a risk of
spurious redefined label errors.

The jump and branch instruction groups from HEX3 are implemented by
functions. A "push label address" instruction must always be followed
immediately by a call to one of the functions that can understand a
relative address: address, branch, jump. The "address" function
(formally "reab") converts the relative address to an absolute
address, which can be stored and used later.

The predefined functions are:

Stack manipulation: drop dup rot pick swap
Arithmetic: + - * / % << >> log
Comparisons: < <= == != >= >
Bitwise logic: & | ^ ~
Memory access: @ = c@ c=
Flow of control, using immediate relative address: branch call
Flow of control, using stored absolute address: call
Address conversion: address
Array support: [] []& []= c[] c[]& c[]=
Access of arguments and variables: arg arg& arg= var var& var=
Function support: enter vars xreturnx xreturn0 xreturn1
Dynamic memory: wsize sbrk / malloc free realloc
System calls: exit in out

- All operations take arguments and return results to the stack.

- Comparisons return 0 or 1.

- All data are words, except for c@, c=, c[], c[]&, c[]=, which
operate on bytes.

- Any user-defined function must start with "enter"; "vars" can be
used straight after "enter" to reserve space for N local variables.

- To return from a function, use one of the "return" functions. "X Y
xreturnx" means return Y values from a function that took X arguments.
The most common cases are Y=0 and Y=1, so "X xreturn0" and "X
xreturn1" are provided.

- Like in C, addresses are byte addresses, so we have to multiply by
wsize when allocating memory with sbrk or malloc.

- "x y []" is equivalent to "x y wsize * + @"

- As always, no forward references to labels are allowed.

As with HEX3 there are two implementations of HEX4. The first one,
hex4a.he, is written in HEX3. The second one, hex4b.he, is written in
HEX4.

./hex3b < hex4a.he > hex4a && chmod +x hex4a
./hex4a < hex4b.he > hex4b && chmod +x hex4b
./hex4b < hex4b.he | diff - hex4b


HEX5: structured programming, at last
-------------------------------------

HEX5 is more like a real structured programming language. There are no
longer any labels; instead there are loops and if...(else)...fi
structures. The syntax of HEX5 can no longer be described with a
regular expression; instead we need a context-free grammar:

program = (hexitem | global | procedure)*
hexitem = hexbyte | "_def" symbol
hexbyte = /'[0-9a-f][0-9a-f]/
global = "var" symbol | "string" symbol string_literal
string_literal = /"([^"]|/.)*"/
procedure = "def" args name "{" vars body "}"
args = symbol*
name = symbol
vars = "var" symbol
body = (number | word | loop | jump | if)*
number = /[0-9]+/
word = symbol
loop = "{" body "}"
jump = "break" | "continue" | "until" | "while"
if = "if" body "fi" | "if" body "else" body "fi"
symbol = /.+/ except ...

Lexical rules:

comment = /#[^n]*n?/
space = /s/
string_literal = /"([^"]|/.)*"/
token = /S+/

The first implementation of HEX5, written in HEX4, is hex5a.he. This
is only a very partial implementation, as it would be quite tedious to
implement all of HEX5 in HEX4. In particular, there are not yet any
named variables or arguments; access to a function's arguments and
local variables is done using the functions from HEX4. Global
variables are implemented with a cunning hack:

./hex4b < hex5a.he > hex5a && chmod +x hex5a

Next came hex5b.he, which can compile itself, as it is written in a
subset of HEX5. In hex5b.he I implemented named arguments and
variables:

./hex5a < hex5b.he > hex5b && chmod +x hex5b
./hex5b < hex5b.he | diff - hex5b

Then I wanted to start using those features for implementing further
features, so I switched to developing hex5c.he, in which I implemented
string constants, "while", "until", "return0" and "return1":

./hex5b < hex5c.he > hex5c && chmod +x hex5c


BCC: a real language
--------------------

All that is needed to turn HEX5 into a tiny structured programming
language is to separate off the first part of the source, where there
is in-line machine code and the "predefined" and library functions are
implemented, into a separate header file. At this point I removed
references to "hex" and called the two files "header.bc" and "bcc.bc".
These two files are concatenated for compilation:

cat header.bc bcc.bc | ./hex5c > bcc && chmod +x bcc

Now bcc can compile itself, of course:

cat header.bc bcc.bc | ./bcc > bcc2 && chmod +x bcc2
mv bcc2 bcc
cat header.bc bcc.bc | ./bcc | diff - bcc

Note that the bcc produced by hex5 might not be identical to the bcc
produced by bcc itself, as I might make some minor improvements to the
code generated by bcc. But the main improvements to be introduced in
bcc are:

- proper error messages to stderr instead of just exit codes
- report undefined symbols
- a dynamic buffer for tokens so there is no limit to their length


What next?
----------

Here are some things that one might want to do with BCC for one's
education and entertainment:

- port it to a different operating system or architecture
(you could compile to Java byte code, for example)

- think of a neater way of handling return values from functions

- implement a compile-time check for stack underflow

- include a non-bogus implementation of malloc, realloc, free

- use an RB-tree for the symbol table so that the compiler does not
take time quadratic in the number of symbols

- think up a way of using BCC to bootstrap GCC ...


Edmund GRIMLEY EVANS , March 2001
Revised: March 2002

 

你可能感兴趣的:(programmer)