264 lines
6.8 KiB
Python
Executable File
264 lines
6.8 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
#
|
|
# Copyright (C) 2015 The Android Open Source Project
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the 'License');
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an 'AS IS' BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
#
|
|
# pylint: disable=bad-indentation,bad-continuation
|
|
import glob
|
|
import os
|
|
import re
|
|
import sys
|
|
|
|
import symbols
|
|
|
|
only_unwanted = False
|
|
if len(sys.argv) > 1:
|
|
if sys.argv[1] in ('-u', '--unwanted'):
|
|
only_unwanted = True
|
|
|
|
toolchain = os.environ['ANDROID_TOOLCHAIN']
|
|
arch = re.sub(r'.*/linux-x86/([^/]+)/.*', r'\1', toolchain)
|
|
if arch == 'aarch64':
|
|
arch = 'arm64'
|
|
|
|
def MangleGlibcNameToBionic(name):
|
|
if name in glibc_to_bionic_names:
|
|
return glibc_to_bionic_names[name]
|
|
return name
|
|
|
|
def GetNdkIgnored(arch): # pylint: disable=redefined-outer-name
|
|
ignored_symbols = set()
|
|
files = glob.glob('%s/ndk/build/tools/unwanted-symbols/%s/*' %
|
|
(os.getenv('ANDROID_BUILD_TOP'), arch))
|
|
for f in files:
|
|
ignored_symbols |= set(open(f, 'r').read().splitlines())
|
|
return ignored_symbols
|
|
|
|
glibc_to_bionic_names = {
|
|
'__res_init': 'res_init',
|
|
'__res_mkquery': 'res_mkquery',
|
|
'__res_query': 'res_query',
|
|
'__res_search': 'res_search',
|
|
'__xpg_basename': '__gnu_basename',
|
|
}
|
|
|
|
glibc = symbols.GetFromSystemSo([
|
|
'libc.so.*',
|
|
'librt.so.*',
|
|
'libpthread.so.*',
|
|
'libresolv.so.*',
|
|
'libm.so.*',
|
|
'libutil.so.*',
|
|
])
|
|
|
|
bionic = symbols.GetFromAndroidSo(['libc.so', 'libm.so'])
|
|
this_dir = os.path.dirname(os.path.realpath(__file__))
|
|
posix = symbols.GetFromTxt(os.path.join(this_dir, 'posix-2013.txt'))
|
|
ndk_ignored = GetNdkIgnored(arch)
|
|
|
|
glibc = set(map(MangleGlibcNameToBionic, glibc))
|
|
|
|
# bionic includes various BSD symbols to ease porting other BSD-licensed code.
|
|
bsd_stuff = set([
|
|
'arc4random',
|
|
'arc4random_buf',
|
|
'arc4random_uniform',
|
|
'basename_r',
|
|
'dirname_r',
|
|
'fgetln',
|
|
'fpurge',
|
|
'funopen',
|
|
'funopen64',
|
|
'gamma_r',
|
|
'gammaf_r',
|
|
'getprogname',
|
|
'setprogname',
|
|
'strlcat',
|
|
'strlcpy',
|
|
'sys_signame',
|
|
'wcslcat',
|
|
'wcslcpy',
|
|
])
|
|
# Some symbols are part of the FORTIFY implementation.
|
|
FORTIFY_stuff = set([
|
|
'__FD_CLR_chk',
|
|
'__FD_ISSET_chk',
|
|
'__FD_SET_chk',
|
|
'__fwrite_chk',
|
|
'__memchr_chk',
|
|
'__memrchr_chk',
|
|
'__pwrite64_chk',
|
|
'__pwrite_chk',
|
|
'__sendto_chk',
|
|
'__stack_chk_guard',
|
|
'__stpncpy_chk2',
|
|
'__strchr_chk',
|
|
'__strlcat_chk',
|
|
'__strlcpy_chk',
|
|
'__strlen_chk',
|
|
'__strncpy_chk2',
|
|
'__strrchr_chk',
|
|
'__umask_chk',
|
|
'__write_chk',
|
|
])
|
|
# Some symbols are used to implement public functions/macros.
|
|
macro_stuff = set([
|
|
'__assert2',
|
|
'__errno',
|
|
'__fe_dfl_env',
|
|
'__get_h_errno',
|
|
'__gnu_strerror_r',
|
|
'__fpclassifyd',
|
|
'__isfinite',
|
|
'__isfinitef',
|
|
'__isfinitel',
|
|
'__isnormal',
|
|
'__isnormalf',
|
|
'__isnormall',
|
|
'__sF',
|
|
'__pthread_cleanup_pop',
|
|
'__pthread_cleanup_push',
|
|
])
|
|
# bionic exposes various Linux features that glibc doesn't.
|
|
linux_stuff = set([
|
|
'getauxval',
|
|
'gettid',
|
|
'pthread_gettid_np',
|
|
'tgkill',
|
|
])
|
|
# Some standard stuff isn't yet in the versions of glibc we're using.
|
|
std_stuff = set([
|
|
'at_quick_exit',
|
|
'c16rtomb',
|
|
'c32rtomb',
|
|
'mbrtoc16',
|
|
'mbrtoc32',
|
|
])
|
|
# These have mangled names in glibc, with a macro taking the "obvious" name.
|
|
weird_stuff = set([
|
|
'fstat',
|
|
'fstat64',
|
|
'fstatat',
|
|
'fstatat64',
|
|
'isfinite',
|
|
'isfinitef',
|
|
'isfinitel',
|
|
'isnormal',
|
|
'isnormalf',
|
|
'isnormall',
|
|
'lstat',
|
|
'lstat64',
|
|
'mknod',
|
|
'mknodat',
|
|
'stat',
|
|
'stat64',
|
|
'optreset',
|
|
'sigsetjmp',
|
|
])
|
|
# These exist in glibc, but under slightly different names (generally one extra
|
|
# or one fewer _). TODO: check against glibc names.
|
|
libresolv_stuff = set([
|
|
'__res_send_setqhook',
|
|
'__res_send_setrhook',
|
|
'_resolv_delete_cache_for_net',
|
|
'_resolv_flush_cache_for_net',
|
|
'_resolv_set_nameservers_for_net',
|
|
'dn_expand',
|
|
'nsdispatch',
|
|
])
|
|
# Implementation details we know we export (and can't get away from).
|
|
known = set([
|
|
'_ctype_',
|
|
'__libc_init',
|
|
])
|
|
# POSIX has some stuff that's unusable in the modern world (a64l) or not
|
|
# actually implemented in glibc unless you count always failing with ENOSYS
|
|
# as being implemented (fattach). Other stuff (fmtmsg) isn't used in any
|
|
# codebase I have access to, internal or external.
|
|
in_posix_and_glibc_but_dead_or_useless = set([
|
|
'a64l', # obsolete
|
|
'confstr', # obsolete
|
|
'endutxent', # no utmp on Android
|
|
'fattach', # <stropts.h> marked obsolescent
|
|
'fdetach', # <stropts.h> marked obsolescent
|
|
'fmtmsg', # unused
|
|
'getdate', # unused
|
|
'getdate_err', # unused
|
|
'gethostid', # obsolete
|
|
'getmsg', # <stropts.h> marked obsolescent
|
|
'getpmsg', # <stropts.h> marked obsolescent
|
|
'getutxent', # no utmp on Android
|
|
'getutxid', # no utmp on Android
|
|
'getutxline', # no utmp on Android
|
|
'isastream', # <stropts.h> marked obsolescent
|
|
'l64a', # obsolete
|
|
'mq_close', # disallowed by SELinux
|
|
'mq_getattr', # disallowed by SELinux
|
|
'mq_notify', # disallowed by SELinux
|
|
'mq_open', # disallowed by SELinux
|
|
'mq_receive', # disallowed by SELinux
|
|
'mq_send', # disallowed by SELinux
|
|
'mq_setattr', # disallowed by SELinux
|
|
'mq_timedreceive', # disallowed by SELinux
|
|
'mq_timedsend', # disallowed by SELinux
|
|
'mq_unlink', # disallowed by SELinux
|
|
'pthread_getconcurrency', # marked obsolescent
|
|
'pthread_setconcurrency', # marked obsolescent
|
|
'putmsg', # <stropts.h> marked obsolescent
|
|
'putpmsg', # <stropts.h> marked obsolescent
|
|
'pututxline', # no utmp on Android
|
|
'shm_open', # disallowed by SELinux
|
|
'shm_unlink', # disallowed by SELinux
|
|
'setutxent', # no utmp on Android
|
|
'sockatmark', # obsolete (https://tools.ietf.org/html/rfc6093)
|
|
'strfmon', # icu4c
|
|
'strfmon_l', # icu4c
|
|
'ulimit', # <ulimit.h> marked obsolescent
|
|
])
|
|
|
|
posix = posix - in_posix_and_glibc_but_dead_or_useless
|
|
glibc = glibc - in_posix_and_glibc_but_dead_or_useless
|
|
|
|
if not only_unwanted:
|
|
#print('glibc:')
|
|
#for symbol in sorted(glibc):
|
|
# print(symbol)
|
|
#print()
|
|
|
|
#print('bionic:')
|
|
#for symbol in sorted(bionic):
|
|
# print(symbol)
|
|
#print()
|
|
|
|
print('in glibc (but not posix) but not bionic:')
|
|
for symbol in sorted((glibc - posix).difference(bionic)):
|
|
print(symbol)
|
|
print()
|
|
|
|
print('in posix (and implemented in glibc) but not bionic:')
|
|
for symbol in sorted((posix.intersection(glibc)).difference(bionic)):
|
|
print(symbol)
|
|
print()
|
|
|
|
print('in bionic but not glibc:')
|
|
|
|
allowed_stuff = (bsd_stuff | FORTIFY_stuff | linux_stuff | macro_stuff |
|
|
std_stuff | weird_stuff | libresolv_stuff | known)
|
|
for symbol in sorted((bionic - allowed_stuff).difference(glibc)):
|
|
if symbol in ndk_ignored:
|
|
symbol += '*'
|
|
print(symbol)
|
|
|
|
sys.exit(0)
|