blob: 6f363e254d7af2c10790eb73c53ee9d72f920c64 [file] [log] [blame]
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -07001#!/usr/bin/python
2#
3# this tool is used to generate the syscall assmbler templates
4# to be placed into arch-x86/syscalls, as well as the content
5# of arch-x86/linux/_syscalls.h
6#
7
The Android Open Source Project4e468ed2008-12-17 18:03:48 -08008import sys, os.path, glob, re, commands, filecmp, shutil
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -07009
10from bionic_utils import *
11
12# set this to 1 if you want to generate thumb stubs
13gen_thumb_stubs = 0
14
15# set this to 1 if you want to generate ARM EABI stubs
16gen_eabi_stubs = 1
17
18# get the root Bionic directory, simply this script's dirname
19#
20bionic_root = find_bionic_root()
21if not bionic_root:
22 print "could not find the Bionic root directory. aborting"
23 sys.exit(1)
24
25if bionic_root[-1] != '/':
26 bionic_root += "/"
27
28print "bionic_root is %s" % bionic_root
29
30# temp directory where we store all intermediate files
31bionic_temp = "/tmp/bionic_gensyscalls/"
32
33# all architectures, update as you see fit
David 'Digit' Turner70b16682012-01-30 17:17:58 +010034all_archs = [ "arm", "x86" ]
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -070035
36def make_dir( path ):
37 if not os.path.exists(path):
38 parent = os.path.dirname(path)
39 if parent:
40 make_dir(parent)
41 os.mkdir(path)
42
43def create_file( relpath ):
44 dir = os.path.dirname( bionic_temp + relpath )
45 make_dir(dir)
46 return open( bionic_temp + relpath, "w" )
47
48# x86 assembler templates for each syscall stub
49#
50
51x86_header = """/* autogenerated by gensyscalls.py */
52#include <sys/linux-syscalls.h>
53
54 .text
55 .type %(fname)s, @function
56 .globl %(fname)s
57 .align 4
58
59%(fname)s:
60"""
61
62x86_registers = [ "%ebx", "%ecx", "%edx", "%esi", "%edi", "%ebp" ]
63
64x86_call = """ movl $%(idname)s, %%eax
65 int $0x80
66 cmpl $-129, %%eax
67 jb 1f
68 negl %%eax
69 pushl %%eax
70 call __set_errno
71 addl $4, %%esp
72 orl $-1, %%eax
731:
74"""
75
76x86_return = """ ret
77"""
78
79# ARM assembler templates for each syscall stub
80#
81arm_header = """/* autogenerated by gensyscalls.py */
Kenny Rootf540c032011-02-17 10:31:30 -080082#include <machine/asm.h>
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -070083#include <sys/linux-syscalls.h>
84
Kenny Rootf540c032011-02-17 10:31:30 -080085ENTRY(%(fname)s)
86"""
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -070087
Kenny Rootf540c032011-02-17 10:31:30 -080088arm_footer = """\
89END(%(fname)s)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -070090"""
91
92arm_call_default = arm_header + """\
93 swi #%(idname)s
94 movs r0, r0
95 bxpl lr
96 b __set_syscall_errno
Kenny Rootf540c032011-02-17 10:31:30 -080097""" + arm_footer
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -070098
99arm_call_long = arm_header + """\
100 .save {r4, r5, lr}
101 stmfd sp!, {r4, r5, lr}
102 ldr r4, [sp, #12]
103 ldr r5, [sp, #16]
104 swi # %(idname)s
105 ldmfd sp!, {r4, r5, lr}
106 movs r0, r0
107 bxpl lr
108 b __set_syscall_errno
Kenny Rootf540c032011-02-17 10:31:30 -0800109""" + arm_footer
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700110
111arm_eabi_call_default = arm_header + """\
112 .save {r4, r7}
113 stmfd sp!, {r4, r7}
114 ldr r7, =%(idname)s
115 swi #0
116 ldmfd sp!, {r4, r7}
117 movs r0, r0
118 bxpl lr
119 b __set_syscall_errno
Kenny Rootf540c032011-02-17 10:31:30 -0800120""" + arm_footer
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700121
122arm_eabi_call_long = arm_header + """\
123 mov ip, sp
124 .save {r4, r5, r6, r7}
125 stmfd sp!, {r4, r5, r6, r7}
126 ldmfd ip, {r4, r5, r6}
127 ldr r7, =%(idname)s
128 swi #0
129 ldmfd sp!, {r4, r5, r6, r7}
130 movs r0, r0
131 bxpl lr
132 b __set_syscall_errno
Kenny Rootf540c032011-02-17 10:31:30 -0800133""" + arm_footer
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700134
135# ARM thumb assembler templates for each syscall stub
136#
137thumb_header = """/* autogenerated by gensyscalls.py */
138 .text
139 .type %(fname)s, #function
140 .globl %(fname)s
141 .align 4
142 .thumb_func
143 .fnstart
144
145#define __thumb__
146#include <sys/linux-syscalls.h>
147
148
149%(fname)s:
150"""
151
152thumb_call_default = thumb_header + """\
153 .save {r7,lr}
154 push {r7,lr}
155 ldr r7, =%(idname)s
156 swi #0
157 tst r0, r0
158 bmi 1f
159 pop {r7,pc}
1601:
161 neg r0, r0
162 ldr r1, =__set_errno
163 blx r1
164 pop {r7,pc}
165 .fnend
166"""
167
168thumb_call_long = thumb_header + """\
169 .save {r4,r5,r7,lr}
170 push {r4,r5,r7,lr}
171 ldr r4, [sp,#16]
172 ldr r5, [sp,#20]
173 ldr r7, =%(idname)s
174 swi #0
175 tst r0, r0
176 bmi 1f
177 pop {r4,r5,r7,pc}
1781:
179 neg r0, r0
180 ldr r1, =__set_errno
181 blx r1
182 pop {r4,r5,r7,pc}
183 .fnend
184"""
185
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100186def param_uses_64bits(param):
187 """Returns True iff a syscall parameter description corresponds
188 to a 64-bit type."""
189 param = param.strip()
190 # First, check that the param type begins with one of the known
191 # 64-bit types.
192 if not ( \
193 param.startswith("int64_t") or param.startswith("uint64_t") or \
194 param.startswith("loff_t") or param.startswith("off64_t") or \
195 param.startswith("long long") or param.startswith("unsigned long long") or
196 param.startswith("signed long long") ):
197 return False
198
199 # Second, check that there is no pointer type here
200 if param.find("*") >= 0:
201 return False
202
203 # Ok
204 return True
205
206def count_arm_param_registers(params):
207 """This function is used to count the number of register used
208 to pass parameters when invoking a thumb or ARM system call.
209 This is because the ARM EABI mandates that 64-bit quantities
210 must be passed in an even+odd register pair. So, for example,
211 something like:
212
213 foo(int fd, off64_t pos)
214
215 would actually need 4 registers:
216 r0 -> int
217 r1 -> unused
218 r2-r3 -> pos
219 """
220 count = 0
221 for param in params:
222 if param_uses_64bits(param):
223 if (count & 1) != 0:
224 count += 1
225 count += 2
226 else:
227 count += 1
228 return count
229
230def count_generic_param_registers(params):
231 count = 0
232 for param in params:
233 if param_uses_64bits(param):
234 count += 2
235 else:
236 count += 1
237 return count
238
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700239class State:
240 def __init__(self):
241 self.old_stubs = []
242 self.new_stubs = []
243 self.other_files = []
244 self.syscalls = []
245
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800246 def x86_genstub(self, fname, numparams, idname):
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700247 t = { "fname" : fname,
248 "idname" : idname }
249
250 result = x86_header % t
251 stack_bias = 4
252 for r in range(numparams):
253 result += " pushl " + x86_registers[r] + "\n"
254 stack_bias += 4
255
256 for r in range(numparams):
257 result += " mov %d(%%esp), %s" % (stack_bias+r*4, x86_registers[r]) + "\n"
258
259 result += x86_call % t
260
261 for r in range(numparams):
262 result += " popl " + x86_registers[numparams-r-1] + "\n"
263
264 result += x86_return
265 return result
266
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800267 def x86_genstub_cid(self, fname, numparams, idname, cid):
268 # We'll ignore numparams here because in reality, if there is a
269 # dispatch call (like a socketcall syscall) there are actually
270 # only 2 arguments to the syscall and 2 regs we have to save:
271 # %ebx <--- Argument 1 - The call id of the needed vectored
272 # syscall (socket, bind, recv, etc)
273 # %ecx <--- Argument 2 - Pointer to the rest of the arguments
274 # from the original function called (socket())
275 t = { "fname" : fname,
276 "idname" : idname }
277
278 result = x86_header % t
279 stack_bias = 4
280
281 # save the regs we need
282 result += " pushl %ebx" + "\n"
283 stack_bias += 4
284 result += " pushl %ecx" + "\n"
285 stack_bias += 4
286
287 # set the call id (%ebx)
288 result += " mov $%d, %%ebx" % (cid) + "\n"
289
290 # set the pointer to the rest of the args into %ecx
291 result += " mov %esp, %ecx" + "\n"
292 result += " addl $%d, %%ecx" % (stack_bias) + "\n"
293
294 # now do the syscall code itself
295 result += x86_call % t
296
297 # now restore the saved regs
298 result += " popl %ecx" + "\n"
299 result += " popl %ebx" + "\n"
300
301 # epilog
302 result += x86_return
303 return result
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700304
305 def arm_genstub(self,fname, flags, idname):
306 t = { "fname" : fname,
307 "idname" : idname }
308 if flags:
309 numargs = int(flags)
310 if numargs > 4:
311 return arm_call_long % t
312 return arm_call_default % t
313
314
315 def arm_eabi_genstub(self,fname, flags, idname):
316 t = { "fname" : fname,
317 "idname" : idname }
318 if flags:
319 numargs = int(flags)
320 if numargs > 4:
321 return arm_eabi_call_long % t
322 return arm_eabi_call_default % t
323
324
325 def thumb_genstub(self,fname, flags, idname):
326 t = { "fname" : fname,
327 "idname" : idname }
328 if flags:
329 numargs = int(flags)
330 if numargs > 4:
331 return thumb_call_long % t
332 return thumb_call_default % t
333
334
335 def process_file(self,input):
336 parser = SysCallsTxtParser()
337 parser.parse_file(input)
338 self.syscalls = parser.syscalls
339 parser = None
340
341 for t in self.syscalls:
342 syscall_func = t["func"]
343 syscall_params = t["params"]
344 syscall_name = t["name"]
345
346 if t["id"] >= 0:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100347 num_regs = count_arm_param_registers(syscall_params)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700348 if gen_thumb_stubs:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100349 t["asm-thumb"] = self.thumb_genstub(syscall_func,num_regs,"__NR_"+syscall_name)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700350 else:
351 if gen_eabi_stubs:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100352 t["asm-arm"] = self.arm_eabi_genstub(syscall_func,num_regs,"__NR_"+syscall_name)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700353 else:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100354 t["asm-arm"] = self.arm_genstub(syscall_func,num_regs,"__NR_"+syscall_name)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700355
356 if t["id2"] >= 0:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100357 num_regs = count_generic_param_registers(syscall_params)
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800358 if t["cid"] >= 0:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100359 t["asm-x86"] = self.x86_genstub_cid(syscall_func, num_regs, "__NR_"+syscall_name, t["cid"])
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800360 else:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100361 t["asm-x86"] = self.x86_genstub(syscall_func, num_regs, "__NR_"+syscall_name)
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800362 elif t["cid"] >= 0:
363 E("cid for dispatch syscalls is only supported for x86 in "
364 "'%s'" % syscall_name)
365 return
366
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700367
368 def gen_NR_syscall(self,fp,name,id):
369 fp.write( "#define __NR_%-25s (__NR_SYSCALL_BASE + %d)\n" % (name,id) )
370
371 # now dump the content of linux/_syscalls.h
372 def gen_linux_syscalls_h(self):
373 path = "include/sys/linux-syscalls.h"
374 D( "generating "+path )
375 fp = create_file( path )
376 fp.write( "/* auto-generated by gensyscalls.py, do not touch */\n" )
Elliott Hughes4a4ed5f2012-05-09 16:34:11 -0700377 fp.write( "#ifndef _BIONIC_LINUX_SYSCALLS_H_\n" )
378 fp.write( "#define _BIONIC_LINUX_SYSCALLS_H_\n\n" )
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700379 fp.write( "#if !defined __ASM_ARM_UNISTD_H && !defined __ASM_I386_UNISTD_H\n" )
380 fp.write( "#if defined __arm__ && !defined __ARM_EABI__ && !defined __thumb__\n" )
381 fp.write( " # define __NR_SYSCALL_BASE 0x900000\n" )
382 fp.write( " #else\n" )
383 fp.write( " # define __NR_SYSCALL_BASE 0\n" )
384 fp.write( " #endif\n\n" )
385
386 # first, all common syscalls
387 for sc in self.syscalls:
388 sc_id = sc["id"]
389 sc_id2 = sc["id2"]
390 sc_name = sc["name"]
391 if sc_id == sc_id2 and sc_id >= 0:
392 self.gen_NR_syscall( fp, sc_name, sc_id )
393
394 # now, all arm-specific syscalls
395 fp.write( "\n#ifdef __arm__\n" );
396 for sc in self.syscalls:
397 sc_id = sc["id"]
398 sc_id2 = sc["id2"]
399 sc_name = sc["name"]
400 if sc_id != sc_id2 and sc_id >= 0:
401 self.gen_NR_syscall( fp, sc_name, sc_id )
402 fp.write( "#endif\n" );
403
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800404 gen_syscalls = {}
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700405 # finally, all i386-specific syscalls
406 fp.write( "\n#ifdef __i386__\n" );
407 for sc in self.syscalls:
408 sc_id = sc["id"]
409 sc_id2 = sc["id2"]
410 sc_name = sc["name"]
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800411 if sc_id != sc_id2 and sc_id2 >= 0 and sc_name not in gen_syscalls:
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700412 self.gen_NR_syscall( fp, sc_name, sc_id2 )
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800413 gen_syscalls[sc_name] = True
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700414 fp.write( "#endif\n" );
415
416 fp.write( "\n#endif\n" )
417 fp.write( "\n#endif /* _BIONIC_LINUX_SYSCALLS_H_ */\n" );
418 fp.close()
419 self.other_files.append( path )
420
421
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700422 # now dump the contents of syscalls.mk
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800423 def gen_arch_syscalls_mk(self, arch):
424 path = "arch-%s/syscalls.mk" % arch
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700425 D( "generating "+path )
426 fp = create_file( path )
427 fp.write( "# auto-generated by gensyscalls.py, do not touch\n" )
428 fp.write( "syscall_src := \n" )
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800429 arch_test = {
430 "arm": lambda x: x.has_key("asm-arm") or x.has_key("asm-thumb"),
Shin-ichiro KAWASAKIce0595d2009-09-01 19:03:06 +0900431 "x86": lambda x: x.has_key("asm-x86"),
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800432 }
433
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700434 for sc in self.syscalls:
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800435 if arch_test[arch](sc):
436 fp.write("syscall_src += arch-%s/syscalls/%s.S\n" %
437 (arch, sc["func"]))
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700438 fp.close()
439 self.other_files.append( path )
440
441 # now generate each syscall stub
442 def gen_syscall_stubs(self):
443 for sc in self.syscalls:
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800444 if sc.has_key("asm-arm") and 'arm' in all_archs:
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700445 fname = "arch-arm/syscalls/%s.S" % sc["func"]
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200446 D2( ">>> generating "+fname )
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700447 fp = create_file( fname )
448 fp.write(sc["asm-arm"])
449 fp.close()
450 self.new_stubs.append( fname )
451
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800452 if sc.has_key("asm-thumb") and 'arm' in all_archs:
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700453 fname = "arch-arm/syscalls/%s.S" % sc["func"]
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200454 D2( ">>> generating "+fname )
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700455 fp = create_file( fname )
456 fp.write(sc["asm-thumb"])
457 fp.close()
458 self.new_stubs.append( fname )
459
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800460 if sc.has_key("asm-x86") and 'x86' in all_archs:
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700461 fname = "arch-x86/syscalls/%s.S" % sc["func"]
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200462 D2( ">>> generating "+fname )
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700463 fp = create_file( fname )
464 fp.write(sc["asm-x86"])
465 fp.close()
466 self.new_stubs.append( fname )
467
468
469 def regenerate(self):
470 D( "scanning for existing architecture-specific stub files" )
471
472 bionic_root_len = len(bionic_root)
473
474 for arch in all_archs:
475 arch_path = bionic_root + "arch-" + arch
476 D( "scanning " + arch_path )
477 files = glob.glob( arch_path + "/syscalls/*.S" )
478 for f in files:
479 self.old_stubs.append( f[bionic_root_len:] )
480
481 D( "found %d stub files" % len(self.old_stubs) )
482
483 if not os.path.exists( bionic_temp ):
484 D( "creating %s" % bionic_temp )
485 os.mkdir( bionic_temp )
486
487# D( "p4 editing source files" )
488# for arch in all_archs:
489# commands.getoutput( "p4 edit " + arch + "/syscalls/*.S " )
490# commands.getoutput( "p4 edit " + arch + "/syscalls.mk" )
491# commands.getoutput( "p4 edit " + bionic_root + "include/sys/linux-syscalls.h" )
492
493 D( "re-generating stubs and support files" )
494
495 self.gen_linux_syscalls_h()
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800496 for arch in all_archs:
497 self.gen_arch_syscalls_mk(arch)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700498 self.gen_syscall_stubs()
499
500 D( "comparing files" )
501 adds = []
502 edits = []
503
504 for stub in self.new_stubs + self.other_files:
505 if not os.path.exists( bionic_root + stub ):
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200506 # new file, git add it
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700507 D( "new file: " + stub)
508 adds.append( bionic_root + stub )
509 shutil.copyfile( bionic_temp + stub, bionic_root + stub )
510
511 elif not filecmp.cmp( bionic_temp + stub, bionic_root + stub ):
512 D( "changed file: " + stub)
513 edits.append( stub )
514
515 deletes = []
516 for stub in self.old_stubs:
517 if not stub in self.new_stubs:
518 D( "deleted file: " + stub)
519 deletes.append( bionic_root + stub )
520
521
522 if adds:
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200523 commands.getoutput("git add " + " ".join(adds))
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700524 if deletes:
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200525 commands.getoutput("git rm " + " ".join(deletes))
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700526 if edits:
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700527 for file in edits:
528 shutil.copyfile( bionic_temp + file, bionic_root + file )
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200529 commands.getoutput("git add " +
530 " ".join((bionic_root + file) for file in edits))
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700531
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200532 commands.getoutput("git add %s%s" % (bionic_root,"SYSCALLS.TXT"))
533
534 if (not adds) and (not deletes) and (not edits):
535 D("no changes detected!")
536 else:
537 D("ready to go!!")
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700538
539D_setlevel(1)
540
541state = State()
542state.process_file(bionic_root+"SYSCALLS.TXT")
543state.regenerate()