forked from brl/citadel
267 lines
8.9 KiB
Python
267 lines
8.9 KiB
Python
|
#!/usr/bin/env python3
|
||
|
#
|
||
|
# Copyright (c) 2012 Intel Corporation
|
||
|
#
|
||
|
# This program is free software; you can redistribute it and/or modify
|
||
|
# it under the terms of the GNU General Public License version 2 as
|
||
|
# published by the Free Software Foundation.
|
||
|
#
|
||
|
# This program is distributed in the hope that it will be useful,
|
||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||
|
# See the GNU General Public License for more details.
|
||
|
#
|
||
|
# You should have received a copy of the GNU General Public License
|
||
|
# along with this program; if not, write to the Free Software
|
||
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||
|
#
|
||
|
# DESCRIPTION
|
||
|
# This script is called by the SDK installer script. It replaces the dynamic
|
||
|
# loader path in all binaries and also fixes the SYSDIR paths/lengths and the
|
||
|
# location of ld.so.cache in the dynamic loader binary
|
||
|
#
|
||
|
# AUTHORS
|
||
|
# Laurentiu Palcu <laurentiu.palcu@intel.com>
|
||
|
#
|
||
|
|
||
|
import struct
|
||
|
import sys
|
||
|
import stat
|
||
|
import os
|
||
|
import re
|
||
|
import errno
|
||
|
|
||
|
if sys.version < '3':
|
||
|
def b(x):
|
||
|
return x
|
||
|
else:
|
||
|
def b(x):
|
||
|
return x.encode(sys.getfilesystemencoding())
|
||
|
|
||
|
old_prefix = re.compile(b("##DEFAULT_INSTALL_DIR##"))
|
||
|
|
||
|
def get_arch():
|
||
|
f.seek(0)
|
||
|
e_ident =f.read(16)
|
||
|
ei_mag0,ei_mag1_3,ei_class = struct.unpack("<B3sB11x", e_ident)
|
||
|
|
||
|
if (ei_mag0 != 0x7f and ei_mag1_3 != "ELF") or ei_class == 0:
|
||
|
return 0
|
||
|
|
||
|
if ei_class == 1:
|
||
|
return 32
|
||
|
elif ei_class == 2:
|
||
|
return 64
|
||
|
|
||
|
def parse_elf_header():
|
||
|
global e_type, e_machine, e_version, e_entry, e_phoff, e_shoff, e_flags,\
|
||
|
e_ehsize, e_phentsize, e_phnum, e_shentsize, e_shnum, e_shstrndx
|
||
|
|
||
|
f.seek(0)
|
||
|
elf_header = f.read(64)
|
||
|
|
||
|
if arch == 32:
|
||
|
# 32bit
|
||
|
hdr_fmt = "<HHILLLIHHHHHH"
|
||
|
hdr_size = 52
|
||
|
else:
|
||
|
# 64bit
|
||
|
hdr_fmt = "<HHIQQQIHHHHHH"
|
||
|
hdr_size = 64
|
||
|
|
||
|
e_type, e_machine, e_version, e_entry, e_phoff, e_shoff, e_flags,\
|
||
|
e_ehsize, e_phentsize, e_phnum, e_shentsize, e_shnum, e_shstrndx =\
|
||
|
struct.unpack(hdr_fmt, elf_header[16:hdr_size])
|
||
|
|
||
|
def change_interpreter(elf_file_name):
|
||
|
if arch == 32:
|
||
|
ph_fmt = "<IIIIIIII"
|
||
|
else:
|
||
|
ph_fmt = "<IIQQQQQQ"
|
||
|
|
||
|
""" look for PT_INTERP section """
|
||
|
for i in range(0,e_phnum):
|
||
|
f.seek(e_phoff + i * e_phentsize)
|
||
|
ph_hdr = f.read(e_phentsize)
|
||
|
if arch == 32:
|
||
|
# 32bit
|
||
|
p_type, p_offset, p_vaddr, p_paddr, p_filesz,\
|
||
|
p_memsz, p_flags, p_align = struct.unpack(ph_fmt, ph_hdr)
|
||
|
else:
|
||
|
# 64bit
|
||
|
p_type, p_flags, p_offset, p_vaddr, p_paddr, \
|
||
|
p_filesz, p_memsz, p_align = struct.unpack(ph_fmt, ph_hdr)
|
||
|
|
||
|
""" change interpreter """
|
||
|
if p_type == 3:
|
||
|
# PT_INTERP section
|
||
|
f.seek(p_offset)
|
||
|
# External SDKs with mixed pre-compiled binaries should not get
|
||
|
# relocated so look for some variant of /lib
|
||
|
fname = f.read(11)
|
||
|
if fname.startswith(b("/lib/")) or fname.startswith(b("/lib64/")) or \
|
||
|
fname.startswith(b("/lib32/")) or fname.startswith(b("/usr/lib32/")) or \
|
||
|
fname.startswith(b("/usr/lib32/")) or fname.startswith(b("/usr/lib64/")):
|
||
|
break
|
||
|
if p_filesz == 0:
|
||
|
break
|
||
|
if (len(new_dl_path) >= p_filesz):
|
||
|
print("ERROR: could not relocate %s, interp size = %i and %i is needed." \
|
||
|
% (elf_file_name, p_memsz, len(new_dl_path) + 1))
|
||
|
break
|
||
|
dl_path = new_dl_path + b("\0") * (p_filesz - len(new_dl_path))
|
||
|
f.seek(p_offset)
|
||
|
f.write(dl_path)
|
||
|
break
|
||
|
|
||
|
def change_dl_sysdirs(elf_file_name):
|
||
|
if arch == 32:
|
||
|
sh_fmt = "<IIIIIIIIII"
|
||
|
else:
|
||
|
sh_fmt = "<IIQQQQIIQQ"
|
||
|
|
||
|
""" read section string table """
|
||
|
f.seek(e_shoff + e_shstrndx * e_shentsize)
|
||
|
sh_hdr = f.read(e_shentsize)
|
||
|
if arch == 32:
|
||
|
sh_offset, sh_size = struct.unpack("<16xII16x", sh_hdr)
|
||
|
else:
|
||
|
sh_offset, sh_size = struct.unpack("<24xQQ24x", sh_hdr)
|
||
|
|
||
|
f.seek(sh_offset)
|
||
|
sh_strtab = f.read(sh_size)
|
||
|
|
||
|
sysdirs = sysdirs_len = ""
|
||
|
|
||
|
""" change ld.so.cache path and default libs path for dynamic loader """
|
||
|
for i in range(0,e_shnum):
|
||
|
f.seek(e_shoff + i * e_shentsize)
|
||
|
sh_hdr = f.read(e_shentsize)
|
||
|
|
||
|
sh_name, sh_type, sh_flags, sh_addr, sh_offset, sh_size, sh_link,\
|
||
|
sh_info, sh_addralign, sh_entsize = struct.unpack(sh_fmt, sh_hdr)
|
||
|
|
||
|
name = sh_strtab[sh_name:sh_strtab.find(b("\0"), sh_name)]
|
||
|
|
||
|
""" look only into SHT_PROGBITS sections """
|
||
|
if sh_type == 1:
|
||
|
f.seek(sh_offset)
|
||
|
""" default library paths cannot be changed on the fly because """
|
||
|
""" the string lengths have to be changed too. """
|
||
|
if name == b(".sysdirs"):
|
||
|
sysdirs = f.read(sh_size)
|
||
|
sysdirs_off = sh_offset
|
||
|
sysdirs_sect_size = sh_size
|
||
|
elif name == b(".sysdirslen"):
|
||
|
sysdirslen = f.read(sh_size)
|
||
|
sysdirslen_off = sh_offset
|
||
|
elif name == b(".ldsocache"):
|
||
|
ldsocache_path = f.read(sh_size)
|
||
|
new_ldsocache_path = old_prefix.sub(new_prefix, ldsocache_path)
|
||
|
new_ldsocache_path = new_ldsocache_path.rstrip(b("\0"))
|
||
|
if (len(new_ldsocache_path) >= sh_size):
|
||
|
print("ERROR: could not relocate %s, .ldsocache section size = %i and %i is needed." \
|
||
|
% (elf_file_name, sh_size, len(new_ldsocache_path)))
|
||
|
sys.exit(-1)
|
||
|
# pad with zeros
|
||
|
new_ldsocache_path += b("\0") * (sh_size - len(new_ldsocache_path))
|
||
|
# write it back
|
||
|
f.seek(sh_offset)
|
||
|
f.write(new_ldsocache_path)
|
||
|
elif name == b(".gccrelocprefix"):
|
||
|
offset = 0
|
||
|
while (offset + 4096) <= sh_size:
|
||
|
path = f.read(4096)
|
||
|
new_path = old_prefix.sub(new_prefix, path)
|
||
|
new_path = new_path.rstrip(b("\0"))
|
||
|
if (len(new_path) >= 4096):
|
||
|
print("ERROR: could not relocate %s, max path size = 4096 and %i is needed." \
|
||
|
% (elf_file_name, len(new_path)))
|
||
|
sys.exit(-1)
|
||
|
# pad with zeros
|
||
|
new_path += b("\0") * (4096 - len(new_path))
|
||
|
#print "Changing %s to %s at %s" % (str(path), str(new_path), str(offset))
|
||
|
# write it back
|
||
|
f.seek(sh_offset + offset)
|
||
|
f.write(new_path)
|
||
|
offset = offset + 4096
|
||
|
if sysdirs != "" and sysdirslen != "":
|
||
|
paths = sysdirs.split(b("\0"))
|
||
|
sysdirs = b("")
|
||
|
sysdirslen = b("")
|
||
|
for path in paths:
|
||
|
""" exit the loop when we encounter first empty string """
|
||
|
if path == b(""):
|
||
|
break
|
||
|
|
||
|
new_path = old_prefix.sub(new_prefix, path)
|
||
|
sysdirs += new_path + b("\0")
|
||
|
|
||
|
if arch == 32:
|
||
|
sysdirslen += struct.pack("<L", len(new_path))
|
||
|
else:
|
||
|
sysdirslen += struct.pack("<Q", len(new_path))
|
||
|
|
||
|
""" pad with zeros """
|
||
|
sysdirs += b("\0") * (sysdirs_sect_size - len(sysdirs))
|
||
|
|
||
|
""" write the sections back """
|
||
|
f.seek(sysdirs_off)
|
||
|
f.write(sysdirs)
|
||
|
f.seek(sysdirslen_off)
|
||
|
f.write(sysdirslen)
|
||
|
|
||
|
# MAIN
|
||
|
if len(sys.argv) < 4:
|
||
|
sys.exit(-1)
|
||
|
|
||
|
# In python > 3, strings may also contain Unicode characters. So, convert
|
||
|
# them to bytes
|
||
|
if sys.version_info < (3,):
|
||
|
new_prefix = sys.argv[1]
|
||
|
new_dl_path = sys.argv[2]
|
||
|
else:
|
||
|
new_prefix = sys.argv[1].encode()
|
||
|
new_dl_path = sys.argv[2].encode()
|
||
|
|
||
|
executables_list = sys.argv[3:]
|
||
|
|
||
|
for e in executables_list:
|
||
|
perms = os.stat(e)[stat.ST_MODE]
|
||
|
if os.access(e, os.W_OK|os.R_OK):
|
||
|
perms = None
|
||
|
else:
|
||
|
os.chmod(e, perms|stat.S_IRWXU)
|
||
|
|
||
|
try:
|
||
|
f = open(e, "r+b")
|
||
|
except IOError:
|
||
|
exctype, ioex = sys.exc_info()[:2]
|
||
|
if ioex.errno == errno.ETXTBSY:
|
||
|
print("Could not open %s. File used by another process.\nPlease "\
|
||
|
"make sure you exit all processes that might use any SDK "\
|
||
|
"binaries." % e)
|
||
|
else:
|
||
|
print("Could not open %s: %s(%d)" % (e, ioex.strerror, ioex.errno))
|
||
|
sys.exit(-1)
|
||
|
|
||
|
# Save old size and do a size check at the end. Just a safety measure.
|
||
|
old_size = os.path.getsize(e)
|
||
|
if old_size >= 64:
|
||
|
arch = get_arch()
|
||
|
if arch:
|
||
|
parse_elf_header()
|
||
|
change_interpreter(e)
|
||
|
change_dl_sysdirs(e)
|
||
|
|
||
|
""" change permissions back """
|
||
|
if perms:
|
||
|
os.chmod(e, perms)
|
||
|
|
||
|
f.close()
|
||
|
|
||
|
if old_size != os.path.getsize(e):
|
||
|
print("New file size for %s is different. Looks like a relocation error!", e)
|
||
|
sys.exit(-1)
|
||
|
|