blob: bfa9fccc359a861955bbd295c1aef34475dbcd99 [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
Shin-ichiro KAWASAKIce0595d2009-09-01 19:03:06 +0900335 def superh_genstub(self, fname, flags, idname):
336 numargs = int(flags)
337 t = { "fname" : fname,
338 "idname" : idname,
339 "numargs" : numargs }
340 superh_call = superh_header
341 if flags:
342 if numargs == 5:
343 superh_call += superh_5args_header
344 if numargs == 6:
345 superh_call += superh_6args_header
346 if numargs == 7:
347 superh_call += superh_7args_header
348 superh_call += superh_call_default
349 return superh_call % t
350
351
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700352 def process_file(self,input):
353 parser = SysCallsTxtParser()
354 parser.parse_file(input)
355 self.syscalls = parser.syscalls
356 parser = None
357
358 for t in self.syscalls:
359 syscall_func = t["func"]
360 syscall_params = t["params"]
361 syscall_name = t["name"]
362
363 if t["id"] >= 0:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100364 num_regs = count_arm_param_registers(syscall_params)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700365 if gen_thumb_stubs:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100366 t["asm-thumb"] = self.thumb_genstub(syscall_func,num_regs,"__NR_"+syscall_name)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700367 else:
368 if gen_eabi_stubs:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100369 t["asm-arm"] = self.arm_eabi_genstub(syscall_func,num_regs,"__NR_"+syscall_name)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700370 else:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100371 t["asm-arm"] = self.arm_genstub(syscall_func,num_regs,"__NR_"+syscall_name)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700372
373 if t["id2"] >= 0:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100374 num_regs = count_generic_param_registers(syscall_params)
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800375 if t["cid"] >= 0:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100376 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 -0800377 else:
David 'Digit' Turner95d751f2010-12-16 16:47:14 +0100378 t["asm-x86"] = self.x86_genstub(syscall_func, num_regs, "__NR_"+syscall_name)
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800379 elif t["cid"] >= 0:
380 E("cid for dispatch syscalls is only supported for x86 in "
381 "'%s'" % syscall_name)
382 return
383
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700384
385 def gen_NR_syscall(self,fp,name,id):
386 fp.write( "#define __NR_%-25s (__NR_SYSCALL_BASE + %d)\n" % (name,id) )
387
388 # now dump the content of linux/_syscalls.h
389 def gen_linux_syscalls_h(self):
390 path = "include/sys/linux-syscalls.h"
391 D( "generating "+path )
392 fp = create_file( path )
393 fp.write( "/* auto-generated by gensyscalls.py, do not touch */\n" )
394 fp.write( "#ifndef _BIONIC_LINUX_SYSCALLS_H_\n\n" )
395 fp.write( "#if !defined __ASM_ARM_UNISTD_H && !defined __ASM_I386_UNISTD_H\n" )
396 fp.write( "#if defined __arm__ && !defined __ARM_EABI__ && !defined __thumb__\n" )
397 fp.write( " # define __NR_SYSCALL_BASE 0x900000\n" )
398 fp.write( " #else\n" )
399 fp.write( " # define __NR_SYSCALL_BASE 0\n" )
400 fp.write( " #endif\n\n" )
401
402 # first, all common syscalls
403 for sc in self.syscalls:
404 sc_id = sc["id"]
405 sc_id2 = sc["id2"]
406 sc_name = sc["name"]
407 if sc_id == sc_id2 and sc_id >= 0:
408 self.gen_NR_syscall( fp, sc_name, sc_id )
409
410 # now, all arm-specific syscalls
411 fp.write( "\n#ifdef __arm__\n" );
412 for sc in self.syscalls:
413 sc_id = sc["id"]
414 sc_id2 = sc["id2"]
415 sc_name = sc["name"]
416 if sc_id != sc_id2 and sc_id >= 0:
417 self.gen_NR_syscall( fp, sc_name, sc_id )
418 fp.write( "#endif\n" );
419
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800420 gen_syscalls = {}
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700421 # finally, all i386-specific syscalls
422 fp.write( "\n#ifdef __i386__\n" );
423 for sc in self.syscalls:
424 sc_id = sc["id"]
425 sc_id2 = sc["id2"]
426 sc_name = sc["name"]
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800427 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 -0700428 self.gen_NR_syscall( fp, sc_name, sc_id2 )
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800429 gen_syscalls[sc_name] = True
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700430 fp.write( "#endif\n" );
431
432 fp.write( "\n#endif\n" )
433 fp.write( "\n#endif /* _BIONIC_LINUX_SYSCALLS_H_ */\n" );
434 fp.close()
435 self.other_files.append( path )
436
437
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700438 # now dump the contents of syscalls.mk
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800439 def gen_arch_syscalls_mk(self, arch):
440 path = "arch-%s/syscalls.mk" % arch
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700441 D( "generating "+path )
442 fp = create_file( path )
443 fp.write( "# auto-generated by gensyscalls.py, do not touch\n" )
444 fp.write( "syscall_src := \n" )
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800445 arch_test = {
446 "arm": lambda x: x.has_key("asm-arm") or x.has_key("asm-thumb"),
Shin-ichiro KAWASAKIce0595d2009-09-01 19:03:06 +0900447 "x86": lambda x: x.has_key("asm-x86"),
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800448 }
449
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700450 for sc in self.syscalls:
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800451 if arch_test[arch](sc):
452 fp.write("syscall_src += arch-%s/syscalls/%s.S\n" %
453 (arch, sc["func"]))
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700454 fp.close()
455 self.other_files.append( path )
456
457 # now generate each syscall stub
458 def gen_syscall_stubs(self):
459 for sc in self.syscalls:
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800460 if sc.has_key("asm-arm") and 'arm' in all_archs:
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700461 fname = "arch-arm/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-arm"])
465 fp.close()
466 self.new_stubs.append( fname )
467
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800468 if sc.has_key("asm-thumb") and 'arm' in all_archs:
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700469 fname = "arch-arm/syscalls/%s.S" % sc["func"]
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200470 D2( ">>> generating "+fname )
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700471 fp = create_file( fname )
472 fp.write(sc["asm-thumb"])
473 fp.close()
474 self.new_stubs.append( fname )
475
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800476 if sc.has_key("asm-x86") and 'x86' in all_archs:
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700477 fname = "arch-x86/syscalls/%s.S" % sc["func"]
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200478 D2( ">>> generating "+fname )
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700479 fp = create_file( fname )
480 fp.write(sc["asm-x86"])
481 fp.close()
482 self.new_stubs.append( fname )
483
484
485 def regenerate(self):
486 D( "scanning for existing architecture-specific stub files" )
487
488 bionic_root_len = len(bionic_root)
489
490 for arch in all_archs:
491 arch_path = bionic_root + "arch-" + arch
492 D( "scanning " + arch_path )
493 files = glob.glob( arch_path + "/syscalls/*.S" )
494 for f in files:
495 self.old_stubs.append( f[bionic_root_len:] )
496
497 D( "found %d stub files" % len(self.old_stubs) )
498
499 if not os.path.exists( bionic_temp ):
500 D( "creating %s" % bionic_temp )
501 os.mkdir( bionic_temp )
502
503# D( "p4 editing source files" )
504# for arch in all_archs:
505# commands.getoutput( "p4 edit " + arch + "/syscalls/*.S " )
506# commands.getoutput( "p4 edit " + arch + "/syscalls.mk" )
507# commands.getoutput( "p4 edit " + bionic_root + "include/sys/linux-syscalls.h" )
508
509 D( "re-generating stubs and support files" )
510
511 self.gen_linux_syscalls_h()
The Android Open Source Project4e468ed2008-12-17 18:03:48 -0800512 for arch in all_archs:
513 self.gen_arch_syscalls_mk(arch)
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700514 self.gen_syscall_stubs()
515
516 D( "comparing files" )
517 adds = []
518 edits = []
519
520 for stub in self.new_stubs + self.other_files:
521 if not os.path.exists( bionic_root + stub ):
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200522 # new file, git add it
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700523 D( "new file: " + stub)
524 adds.append( bionic_root + stub )
525 shutil.copyfile( bionic_temp + stub, bionic_root + stub )
526
527 elif not filecmp.cmp( bionic_temp + stub, bionic_root + stub ):
528 D( "changed file: " + stub)
529 edits.append( stub )
530
531 deletes = []
532 for stub in self.old_stubs:
533 if not stub in self.new_stubs:
534 D( "deleted file: " + stub)
535 deletes.append( bionic_root + stub )
536
537
538 if adds:
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200539 commands.getoutput("git add " + " ".join(adds))
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700540 if deletes:
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200541 commands.getoutput("git rm " + " ".join(deletes))
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700542 if edits:
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700543 for file in edits:
544 shutil.copyfile( bionic_temp + file, bionic_root + file )
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200545 commands.getoutput("git add " +
546 " ".join((bionic_root + file) for file in edits))
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700547
David 'Digit' Turnerfc269312010-10-11 22:11:06 +0200548 commands.getoutput("git add %s%s" % (bionic_root,"SYSCALLS.TXT"))
549
550 if (not adds) and (not deletes) and (not edits):
551 D("no changes detected!")
552 else:
553 D("ready to go!!")
The Android Open Source Projecta27d2ba2008-10-21 07:00:00 -0700554
555D_setlevel(1)
556
557state = State()
558state.process_file(bionic_root+"SYSCALLS.TXT")
559state.regenerate()