xref: /aosp_15_r20/bionic/libc/kernel/tools/update_all.py (revision 8d67ca893c1523eb926b9080dbe4e2ffd2a27ba1)
1*8d67ca89SAndroid Build Coastguard Worker#!/usr/bin/env python3
2*8d67ca89SAndroid Build Coastguard Worker#
3*8d67ca89SAndroid Build Coastguard Workerimport sys, cpp, kernel, glob, os, re, getopt, clean_header, shutil
4*8d67ca89SAndroid Build Coastguard Workerfrom defaults import *
5*8d67ca89SAndroid Build Coastguard Workerfrom utils import *
6*8d67ca89SAndroid Build Coastguard Worker
7*8d67ca89SAndroid Build Coastguard Workerdef Usage():
8*8d67ca89SAndroid Build Coastguard Worker    print("""\
9*8d67ca89SAndroid Build Coastguard Worker  usage: %(progname)s [kernel-original-path] [kernel-modified-path]
10*8d67ca89SAndroid Build Coastguard Worker
11*8d67ca89SAndroid Build Coastguard Worker    this program is used to update all the auto-generated clean headers
12*8d67ca89SAndroid Build Coastguard Worker    used by the Bionic C library. it assumes the following:
13*8d67ca89SAndroid Build Coastguard Worker
14*8d67ca89SAndroid Build Coastguard Worker      - a set of source kernel headers is located in
15*8d67ca89SAndroid Build Coastguard Worker        'external/kernel-headers/original', relative to the current
16*8d67ca89SAndroid Build Coastguard Worker        android tree
17*8d67ca89SAndroid Build Coastguard Worker
18*8d67ca89SAndroid Build Coastguard Worker      - a set of manually modified kernel header files located in
19*8d67ca89SAndroid Build Coastguard Worker        'external/kernel-headers/modified', relative to the current
20*8d67ca89SAndroid Build Coastguard Worker        android tree
21*8d67ca89SAndroid Build Coastguard Worker
22*8d67ca89SAndroid Build Coastguard Worker      - the clean headers will be placed in 'bionic/libc/kernel/arch-<arch>/asm',
23*8d67ca89SAndroid Build Coastguard Worker        'bionic/libc/kernel/android', etc..
24*8d67ca89SAndroid Build Coastguard Worker""" % { "progname" : os.path.basename(sys.argv[0]) })
25*8d67ca89SAndroid Build Coastguard Worker    sys.exit(0)
26*8d67ca89SAndroid Build Coastguard Worker
27*8d67ca89SAndroid Build Coastguard Workerdef ProcessFiles(updater, original_dir, modified_dir, src_rel_dir, update_rel_dir):
28*8d67ca89SAndroid Build Coastguard Worker    # Delete the old headers before updating to the new headers.
29*8d67ca89SAndroid Build Coastguard Worker    update_dir = os.path.join(get_kernel_dir(), update_rel_dir)
30*8d67ca89SAndroid Build Coastguard Worker    for root, dirs, files in os.walk(update_dir, topdown=True):
31*8d67ca89SAndroid Build Coastguard Worker        for entry in files:
32*8d67ca89SAndroid Build Coastguard Worker            # BUILD is a special file that needs to be preserved.
33*8d67ca89SAndroid Build Coastguard Worker            if entry == "BUILD":
34*8d67ca89SAndroid Build Coastguard Worker                continue
35*8d67ca89SAndroid Build Coastguard Worker            os.remove(os.path.join(root, entry))
36*8d67ca89SAndroid Build Coastguard Worker        for entry in dirs:
37*8d67ca89SAndroid Build Coastguard Worker            shutil.rmtree(os.path.join(root, entry))
38*8d67ca89SAndroid Build Coastguard Worker
39*8d67ca89SAndroid Build Coastguard Worker    src_dir = os.path.normpath(os.path.join(original_dir, src_rel_dir))
40*8d67ca89SAndroid Build Coastguard Worker    src_dir_len = len(src_dir) + 1
41*8d67ca89SAndroid Build Coastguard Worker    mod_src_dir = os.path.join(modified_dir, src_rel_dir)
42*8d67ca89SAndroid Build Coastguard Worker    update_dir = os.path.join(get_kernel_dir(), update_rel_dir)
43*8d67ca89SAndroid Build Coastguard Worker
44*8d67ca89SAndroid Build Coastguard Worker    kernel_dir = get_kernel_dir()
45*8d67ca89SAndroid Build Coastguard Worker    for root, _, files in os.walk(src_dir):
46*8d67ca89SAndroid Build Coastguard Worker        for file in sorted(files):
47*8d67ca89SAndroid Build Coastguard Worker            _, ext = os.path.splitext(file)
48*8d67ca89SAndroid Build Coastguard Worker            if ext != ".h":
49*8d67ca89SAndroid Build Coastguard Worker                continue
50*8d67ca89SAndroid Build Coastguard Worker            src_file = os.path.normpath(os.path.join(root, file))
51*8d67ca89SAndroid Build Coastguard Worker            rel_path = src_file[src_dir_len:]
52*8d67ca89SAndroid Build Coastguard Worker            # Check to see if there is a modified header to use instead.
53*8d67ca89SAndroid Build Coastguard Worker            if os.path.exists(os.path.join(mod_src_dir, rel_path)):
54*8d67ca89SAndroid Build Coastguard Worker                src_file = os.path.join(mod_src_dir, rel_path)
55*8d67ca89SAndroid Build Coastguard Worker                src_str = os.path.join("<modified>", src_rel_dir, rel_path)
56*8d67ca89SAndroid Build Coastguard Worker            else:
57*8d67ca89SAndroid Build Coastguard Worker                src_str = os.path.join("<original>", src_rel_dir, rel_path)
58*8d67ca89SAndroid Build Coastguard Worker            dst_file = os.path.join(update_dir, rel_path)
59*8d67ca89SAndroid Build Coastguard Worker            new_data = clean_header.cleanupFile(dst_file, src_file, rel_path)
60*8d67ca89SAndroid Build Coastguard Worker            if not new_data:
61*8d67ca89SAndroid Build Coastguard Worker                continue
62*8d67ca89SAndroid Build Coastguard Worker            updater.readFile(dst_file)
63*8d67ca89SAndroid Build Coastguard Worker            ret_val = updater.editFile(dst_file, new_data)
64*8d67ca89SAndroid Build Coastguard Worker            if ret_val == 0:
65*8d67ca89SAndroid Build Coastguard Worker                state = "unchanged"
66*8d67ca89SAndroid Build Coastguard Worker            elif ret_val == 1:
67*8d67ca89SAndroid Build Coastguard Worker                state = "edited"
68*8d67ca89SAndroid Build Coastguard Worker            else:
69*8d67ca89SAndroid Build Coastguard Worker                state = "added"
70*8d67ca89SAndroid Build Coastguard Worker            update_path = os.path.join(update_rel_dir, rel_path)
71*8d67ca89SAndroid Build Coastguard Worker            print("cleaning %s -> %s (%s)" % (src_str, update_path, state))
72*8d67ca89SAndroid Build Coastguard Worker
73*8d67ca89SAndroid Build Coastguard Worker
74*8d67ca89SAndroid Build Coastguard Worker# This lets us support regular system calls like __NR_write and also weird
75*8d67ca89SAndroid Build Coastguard Worker# ones like __ARM_NR_cacheflush, where the NR doesn't come at the start.
76*8d67ca89SAndroid Build Coastguard Workerdef make__NR_name(name):
77*8d67ca89SAndroid Build Coastguard Worker    if name.startswith('__ARM_NR_'):
78*8d67ca89SAndroid Build Coastguard Worker        return name
79*8d67ca89SAndroid Build Coastguard Worker    else:
80*8d67ca89SAndroid Build Coastguard Worker        return '__NR_%s' % (name)
81*8d67ca89SAndroid Build Coastguard Worker
82*8d67ca89SAndroid Build Coastguard Worker
83*8d67ca89SAndroid Build Coastguard Worker# Scan Linux kernel asm/unistd.h files containing __NR_* constants
84*8d67ca89SAndroid Build Coastguard Worker# and write out equivalent SYS_* constants for glibc source compatibility.
85*8d67ca89SAndroid Build Coastguard Workerdef GenerateGlibcSyscallsHeader(updater):
86*8d67ca89SAndroid Build Coastguard Worker    libc_root = '%s/bionic/libc/' % os.environ['ANDROID_BUILD_TOP']
87*8d67ca89SAndroid Build Coastguard Worker
88*8d67ca89SAndroid Build Coastguard Worker    # Collect the set of all syscalls for all architectures.
89*8d67ca89SAndroid Build Coastguard Worker    syscalls = set()
90*8d67ca89SAndroid Build Coastguard Worker    pattern = re.compile(r'^\s*#\s*define\s*__NR_([a-z_]\S+)')
91*8d67ca89SAndroid Build Coastguard Worker    for unistd_h in glob.glob('%s/kernel/uapi/asm-*/asm/unistd*.h' % libc_root):
92*8d67ca89SAndroid Build Coastguard Worker        for line in open(unistd_h):
93*8d67ca89SAndroid Build Coastguard Worker            m = re.search(pattern, line)
94*8d67ca89SAndroid Build Coastguard Worker            if m:
95*8d67ca89SAndroid Build Coastguard Worker                nr_name = m.group(1)
96*8d67ca89SAndroid Build Coastguard Worker                if 'reserved' not in nr_name and 'unused' not in nr_name:
97*8d67ca89SAndroid Build Coastguard Worker                    syscalls.add(nr_name)
98*8d67ca89SAndroid Build Coastguard Worker
99*8d67ca89SAndroid Build Coastguard Worker    # Create a single file listing them all.
100*8d67ca89SAndroid Build Coastguard Worker    # Note that the input files include #if trickery, so even for a single
101*8d67ca89SAndroid Build Coastguard Worker    # architecture we don't know exactly which ones are available.
102*8d67ca89SAndroid Build Coastguard Worker    # https://b.corp.google.com/issues/37110151
103*8d67ca89SAndroid Build Coastguard Worker    content = '/* Generated file. Do not edit. */\n'
104*8d67ca89SAndroid Build Coastguard Worker    content += '#pragma once\n'
105*8d67ca89SAndroid Build Coastguard Worker
106*8d67ca89SAndroid Build Coastguard Worker    for syscall in sorted(syscalls):
107*8d67ca89SAndroid Build Coastguard Worker        nr_name = make__NR_name(syscall)
108*8d67ca89SAndroid Build Coastguard Worker        content += '#if defined(%s)\n' % nr_name
109*8d67ca89SAndroid Build Coastguard Worker        content += '  #define SYS_%s %s\n' % (syscall, nr_name)
110*8d67ca89SAndroid Build Coastguard Worker        content += '#endif\n'
111*8d67ca89SAndroid Build Coastguard Worker
112*8d67ca89SAndroid Build Coastguard Worker    syscall_file = os.path.join(libc_root, 'include/bits/glibc-syscalls.h')
113*8d67ca89SAndroid Build Coastguard Worker    updater.readFile(syscall_file)
114*8d67ca89SAndroid Build Coastguard Worker    updater.editFile(syscall_file, content)
115*8d67ca89SAndroid Build Coastguard Worker
116*8d67ca89SAndroid Build Coastguard Worker
117*8d67ca89SAndroid Build Coastguard Workertry:
118*8d67ca89SAndroid Build Coastguard Worker    optlist, args = getopt.getopt(sys.argv[1:], '')
119*8d67ca89SAndroid Build Coastguard Workerexcept:
120*8d67ca89SAndroid Build Coastguard Worker    # Unrecognized option
121*8d67ca89SAndroid Build Coastguard Worker    sys.stderr.write("error: unrecognized option\n")
122*8d67ca89SAndroid Build Coastguard Worker    Usage()
123*8d67ca89SAndroid Build Coastguard Worker
124*8d67ca89SAndroid Build Coastguard Workerif len(optlist) > 0 or len(args) > 2:
125*8d67ca89SAndroid Build Coastguard Worker    Usage()
126*8d67ca89SAndroid Build Coastguard Worker
127*8d67ca89SAndroid Build Coastguard Workerif len(args) > 0:
128*8d67ca89SAndroid Build Coastguard Worker    original_dir = args[0]
129*8d67ca89SAndroid Build Coastguard Workerelse:
130*8d67ca89SAndroid Build Coastguard Worker    original_dir = get_kernel_headers_original_dir()
131*8d67ca89SAndroid Build Coastguard Worker
132*8d67ca89SAndroid Build Coastguard Workerif len(args) > 1:
133*8d67ca89SAndroid Build Coastguard Worker    modified_dir = args[1]
134*8d67ca89SAndroid Build Coastguard Workerelse:
135*8d67ca89SAndroid Build Coastguard Worker    modified_dir = get_kernel_headers_modified_dir()
136*8d67ca89SAndroid Build Coastguard Worker
137*8d67ca89SAndroid Build Coastguard Workerif not os.path.isdir(original_dir):
138*8d67ca89SAndroid Build Coastguard Worker    panic("The kernel directory %s is not a directory\n" % original_dir)
139*8d67ca89SAndroid Build Coastguard Worker
140*8d67ca89SAndroid Build Coastguard Workerif not os.path.isdir(modified_dir):
141*8d67ca89SAndroid Build Coastguard Worker    panic("The kernel modified directory %s is not a directory\n" % modified_dir)
142*8d67ca89SAndroid Build Coastguard Worker
143*8d67ca89SAndroid Build Coastguard Workerupdater = BatchFileUpdater()
144*8d67ca89SAndroid Build Coastguard Worker
145*8d67ca89SAndroid Build Coastguard Worker# Process the original uapi headers first.
146*8d67ca89SAndroid Build Coastguard WorkerProcessFiles(updater, original_dir, modified_dir, "uapi", "uapi"),
147*8d67ca89SAndroid Build Coastguard Worker
148*8d67ca89SAndroid Build Coastguard Worker# Now process the special files.
149*8d67ca89SAndroid Build Coastguard WorkerProcessFiles(updater, original_dir, modified_dir, "scsi", os.path.join("android", "scsi", "scsi"))
150*8d67ca89SAndroid Build Coastguard Worker
151*8d67ca89SAndroid Build Coastguard Worker# Copy all of the files.
152*8d67ca89SAndroid Build Coastguard Workerupdater.updateFiles()
153*8d67ca89SAndroid Build Coastguard Worker
154*8d67ca89SAndroid Build Coastguard Worker# Now re-generate the <bits/glibc-syscalls.h> from the new uapi headers.
155*8d67ca89SAndroid Build Coastguard Workerupdater = BatchFileUpdater()
156*8d67ca89SAndroid Build Coastguard WorkerGenerateGlibcSyscallsHeader(updater)
157*8d67ca89SAndroid Build Coastguard Workerupdater.updateFiles()
158