- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我有一个 Linux 二进制文件,没有源代码,可以在一台机器上运行,我想制作一个独立的包,可以在同一架构的不同机器上运行。实现这一目标的方法是什么?
在我的例子中,两台机器都有相同的架构,相同的 Ubuntu 内核,但目标机器没有 make
并且在 /lib
下有错误的文件版本和/usr
我的一个想法是使用 chroot
并重新创建二进制文件使用的文件系统的子集,可能使用 strace
来确定它需要什么。是否已经有可以执行此操作的工具?
为了后代,下面是我如何确定一个进程打开了哪些文件
#!/usr/bin/python
# source of trace_fileopen.py
# Runs command and prints all files that have been successfully opened with mode O_RDONLY
# example: trace_fileopen.py ls -l
import re, sys, subprocess, os
if __name__=='__main__':
strace_fn = '/tmp/strace.out'
strace_re = re.compile(r'([^(]+?)\((.*)\)\s*=\s*(\S+?)\s+(.*)$')
cmd = sys.argv[1]
nowhere = open('/dev/null','w')#
p = subprocess.Popen(['strace','-o', strace_fn]+sys.argv[1:], stdout=nowhere, stderr=nowhere)
sts = os.waitpid(p.pid, 0)[1]
output = []
for line in open(strace_fn):
# ignore lines like --- SIGCHLD (Child exited) @ 0 (0) ---
if not strace_re.match(line):
continue
(function,args,returnval,msg) = strace_re.findall(line)[0]
if function=='open' and returnval!='-1':
(fname,mode)=args.split(',',1)
if mode.strip()=='O_RDONLY':
if fname.startswith('"') and fname.endswith('"') and len(fname)>=2:
fname = fname[1:-1]
output.append(fname)
prev_line = ""
for line in sorted(output):
if line==prev_line:
continue
print line
prev_line = line
更新LD_LIBRARY_PATH
解决方案的问题是 /lib
被硬编码到解释器中并优先于 LD_LIBRARY_PATH
,因此 native 版本将首先加载。解释器被硬编码到二进制文件中。一种方法可能是修补解释器并将二进制文件作为 patched_interpreter mycommandline
问题是当 mycommandline
以 java
开头时,这不会工作,因为 Java 设置 LD_LIBRARY_PATH
并重新启动自身,它求助于旧的解释器。对我有用的解决方案是在文本编辑器中打开二进制文件,找到解释器 (/lib/ld-linux-x86-64.so.2
),并将其替换为相同长度补丁解释器的路径
最佳答案
正如其他人所提到的,静态链接是一种选择。除了与 glibc 的静态链接在每个版本中都会出现一些问题(抱歉,没有引用资料;只是我的经验)。
您的 chroot
想法可能有点矫枉过正。
据我所知,大多数商业产品使用的解决方案是让它们的“应用程序”成为一个设置 LD_LIBRARY_PATH
的 shell 脚本,然后运行实际的可执行文件。沿着这些线的东西:
#!/bin/sh
here=`dirname "$0"`
export LD_LIBRARY_PATH="$here"/lib
exec "$here"/bin/my_app "$@"
然后您只需将所有相关 .so 文件的副本转储到 lib/
下,将您的可执行文件放在 bin/
下,将脚本放在 中。
,并运送整棵树。
(为了具有生产值(value),如果 LD_LIBRARY_PATH
不为空,请将 "$here"/lib
适本地放在前面,等等)
[编辑,配合你的更新]
我认为您可能对什么是硬编码什么不是硬编码感到困惑。 ld-linux-x86-64.so.2
是动态链接器本身;并且您是正确的,它的路径被硬编码到 ELF header 中。但是其他库不是硬编码的;它们由动态链接器搜索,动态链接器将遵循 LD_LIBRARY_PATH
。
如果你真的需要一个不同的 ld-linux.so,而不是修补 ELF 头,只需运行动态链接器本身:
/path/to/my-ld-linux.so my_program <args>
这将使用您的链接器而不是 ELF header 中列出的链接器。
修补可执行文件本身是邪恶的。请考虑在您继续前进后必须维护您的东西的可怜人......没有人会期望您手工破解 ELF header 。 任何人都可以阅读 shell 脚本在做什么。
只要我的 0.02 美元。
关于linux - 如何使 Unix 二进制文件自包含?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/6701650/
正如标题所暗示的那样,我无法弄清楚 Unix 内核如何将逻辑文件偏移量转换为逻辑块号,然后从 i-node 中检索它。 作为引用,我要求对 Maurice J. Bach 在“UNIX 操作系统的设计
关闭。这个问题是off-topic .它目前不接受答案。 想改善这个问题吗? Update the question所以它是 on-topic对于堆栈溢出。 8年前关闭。 Improve this q
我在互联网上做了一些研究,但仍然很困惑。 UNIX 时间是像 GMT/UTC 那样的通用时间还是像本地时间一样因地而异? 我知道 UNIX 时间是从 1970 年 1 月 1 日格林威治标准时间 00
您如何评估 Unix 系统管理员。 Unix 管理员应该具备哪些编程能力? 最佳答案 我用于快速过滤器的一些: 什么是 fork 炸弹,它是好是坏? 给我一个单行命令,计算日志文件中有多少行从昨天的日
谁能告诉我字典文本文件在 UNIX 系统上的位置?或者我在哪里可以获得一个好的字典文本文件?我目前一直在使用来自 SUN 的文本文件,但它包含不带句点的缩写(否则我可以删除它们)。有人能指出我正确的方
在我的 Ubuntu 12 vps 上,我正在运行一个完整的比特币节点。当我第一次启动它时,它使用了大约 700mb 的内存。如果我 24 小时后回来 (free -m) 将如下所示: total
我想编写一个 unix/linux 程序,它将使用一个配置文件。 我的问题是,我应该把文件的位置放在哪里? 我可以将位置(如 /etc )“硬编码”到程序本身中。 但是,我希望它,如果没有权限的用户可
在UNIX脚本编程中,cat是可以将2个文件组合在一起的命令: cat file1 file2 > file3 通过合并前两个生成第三个。 另外,cat可以与管道一起使用: cat file1 | t
我有一个通过 ssh 连接到外部机器的终端,并且有一个进程在其中运行。 是否可以将执行移到后台,以便我可以关闭 ssh 连接而无需终止它?如果是这样怎么办? 最佳答案 按 control + Z,这将
我正在试验我自己的 BSD 或 Linux 发行版。我想以对最终用户有意义的方式组织系统文件。我希望他们能够访问系统,而不会出现 *nixes 留下的所有文件困惑。 有没有办法在不丢失动态链接的情况下
这条评论让我感到困惑:“kill -l 通常会列出所有信号”。我认为信号意味着量化的能量。 [已添加] 请澄清 Unix 中的(计算)信号和物理信号。它们是完全不同的概念吗? [已添加] 范式之间是否
fuser 命令让我知道哪些进程正在使用文件或目录。 我正在寻找相反的命令:让我知道进程正在使用哪些文件。 更新 忘了说它是针对 Solaris 系统的。 最佳答案 lsof -p 来自 here
如果我有一个叫做“orange”的词,我如何将它拆分成单独的字符。 我的输出应该是: o r a n g e 最佳答案 echo orange | fold -w 1 输出 o r a n g e 关
和有什么区别工作和一个 流程在 Unix 中?你能举个例子吗? 最佳答案 作业是由 shell 启动的进程。 shell 在作业表中跟踪这些。作业命令显示事件后台进程的列表。他们得到一个 jobspe
unix 如何处理带空格和参数的完整路径名? 在 Windows 中,我们引用路径并在其后添加命令行参数,在 unix 中如何? "c:\foo folder with space\foo.exe"
我必须合并具有相同标题的多个CSV文件。 我必须保留第一个文件的 header ,并删除所有其他文件的 header ,然后合并它们并创建一个主文件。 文件1: Id,city,name ,locat
我需要在两个字段上加入两个文件。但是,即使连接失败,我也应该检索文件 1 中的所有值,就像左外连接一样。 文件 1: 01|a|jack|d 02|b|ron|c 03|d|tom|e 文件2: 01
在 Solaris, HP-UX 上获取进程大小的正确方法是什么?和 AIX ?我们应该使用 top或 ps -o vsz或者是其他东西? 最佳答案 vsize的确切定义, rss , rprvt ,
我在文件上使用了“touch”,更新了文件的时间戳,但父目录的时间戳没有改变。但是,(如预期)当我在父目录中创建新文件时,该目录的时间戳确实发生了变化。 类 UNIX 操作系统(特别是 AIX)使用什
一般来说,当我们从多个进程向 UNIX 中的文件追加内容时,我们可以认为什么是理所当然的?是否有可能丢失数据(一个进程覆盖另一个进程的更改)?数据有可能被破坏吗? (例如,每个进程都将每个追加一行追加
我是一名优秀的程序员,十分优秀!