-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathPorting
155 lines (105 loc) · 5.29 KB
/
Porting
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
PORTING SUBC TO DIFFERENT PLATFORMS
(1) Create a new target description file
Copy one of the existing cg*.c files from the src/targets/cg/
directory to create a new target description. The target
description will contain the instructions that will be
emitted by the code generator. Also copy the corresponding
cg*.h file. For example, to create a "CPU" back-end:
cp src/targets/cg/cg386.c src/targets/cg/cgCPU.c
cp src/targets/cg/cg386.h src/targets/cg/cgCPU.h
(2) Select the proper machine word size
Edit the newly created header file and change its BPW constant
to the *natural* machine word size on the new target. This
would be 32 bits for a 32-bit processor, 64 bits for a 64-bit
processor, etc. Note that the word size selected here must be
large enough to hold an arbitrary address (a pointer) on the
target. Incidentally, the selected size will be sizeof(int) on
the new back-end. Also change the CPU constant to the name of
your CPU.
vi src/targets/cg/cgCPU.h
(3) Create an environment description for your OS
The SubC compiler uses some external resources that are part
of your operating system, like the assembler, linker, and
maybe the system's C library. These environment-specific
information should be contained in a separate directory in
target/ that is named after the targeted OS and the processor:
mkdir src/targets/OS-CPU
That directory must contain a file that is also named after
the OS and processor, but with a "sys-" prefix (e.g.
sys-linux-386.h). This file has to contain the name of the OS
and the full paths or templates for accessing any external
resources. Best use one of the existing files for a template.
vi src/targets/OS-CPU/sys-OS-CPU.h
(4) Create a target description
This is one of the hard parts. You will have to replace the
code emitted by each function in the new target description
file to generate code for your new platform. If you are an
experienced compiler writer and you are familiar with one of
the existing targets, you may be able to guess the proper
sequences. Otherwise, you probably need a copy of the book
"Practical Compiler Construction"[1] now, which explains the
target description format in detail.
[1] See the README file.
For instance, to adapt the 386 target to the x86-64 processor,
this step basically involves (a) replacing register names (%eax
to %rax, etc), (b) replacing instruction suffixes (movl->movq,
etc), (c) replacing ".long" by ".quad", and (d) adjusting all
values that deal with word size, i.e. adding multiples of 8
instead of 4 to registers in pointer arithmetics, shifting by
3 instead of 2, etc.
Of course, other processors will differ from the 386 by a far
greater margin, so this step can be quite a piece of work.
The conclusion part of the book contains some practical tips
for approaching this task.
(5) Create an OS-specific C startup file
Copy the crt0-*.s file from one of the existing
src/targets/OS-CPU/ directories to create a new
startup file, e.g.:
cp src/targets/crt0-freebsd-386.s \
src/targets/crt0-OS-CPU.s
Again, OS is your OS, and CPU is the new CPU being
targeted.
(6) Create the OS interface
This step is a mess and requires some decent in-depth knowledge
of the target platform, i.e. its internal command line argument
format, environment location, its calling conventions, system
call interface, etc.
Again, if you are really familiar with one of the supported
platforms, you may be able to rewrite the individual functions.
Otherwise, this is a good time to pick up a copy of "Practical
Compiler Construction", which also explains this part in depth.
You may run into a lot of trouble at this point. Be patient,
this is to be expected.
vi src/targets/OS-CPU/crt0-OS-CPU.s
There is a useful tool called "sys" in the tests/ directory of
the SubC directory tree. When compiled with SubC, it can be used
to test the system call wrapper, program argument passing and
environment access in the crt0 module.
./scc0 -o sys ../tests/sys.c
(7) Compile, test, repeat
A lot of things *will* go wrong. Do not get discouraged! This
is normal. First make the compiler emit syntactically correct
assembler code. Then make it compile the library. Then test
some small programs. Then compile the compiler with itself.
Once it passes the triple test (make test), the compiler can
be considered to be stable.
Pointer arithmetics can be particularly nasty, so there is an
ad-hoc pointer arithmetics test suite included in the SubC
archive. Just run 'make ptest' in src/. If it does not print
anything, fine!
The system call interface can be ad-hoc tested by running
"make systest". It is pretty likely to blow up at some point
when there is something wrong with the crt0 module.
In a similar way, most parts of the C library can be tested
by running "make libtest".
Signal handling is particurlarly nasty and almost impossible
to get right these days. "make sigtest" will help you find out
if your signal() implementation blows up.
(8) Add your back-end to the configure script
Have a look at the configure script in the root of the SubC
source tree and add your configuration. It is just a simple,
human readable script containing no magic whatsoever, so this
should be a piece of cake.
vi configure
(9) Optionally: send your patches to the author
Thank you!