修改大型二进制库内部函数名的bash 脚本及其解释

1.大型lib库重命名函数名字

迭代的方式对于大型二进制库改名字的功能脚本

会将源文件中的函数 add_(...) 修改成 nubia_add_(...) 的方式来调用

此脚本分配来修改,可以避免突破资源限制 ulimit -a;

保存为:redef_func_name_01.sh

chmod u+x redef_func_name.sh

如果使能 echo -L "$new"

那么,其中api_symbols=( )中要写入本lib的公共的暴露出来的api函数名,否则都不能被调用。

bash 复制代码
#!/bin/bash


apitxt="./funclist.txt"
prefix_name="nubia_"

api_symbols=(  )

#1. 将 库文件的符号导出到本地文本文件
rm -rf ${apitxt}
nm $1 > ${apitxt}

#文本文件总行数
N=$(sed -n '$=' $apitxt)

#sed -n "300, 400" filename 注意p;读出文件的 第300行到第400行


for ((i=1; i<=$N; i+=1000))
do
j=$[i + 999]

#debug
#seg=($(sed -n "${i}, ${j}p" ${apitxt}))
#echo -e ${seg[@]}
#echo $i $j
#取出i行到j行,只保留T类型后面的文本,即库中源代码实现的函数名字,保存进internal_symbols中
internal_symbols=($(
    sed -n "${i}, ${j}p" $apitxt | sed '/^[0-9]\+ T /!d; s///' |
    sort | comm -13 <(printf "%s\n" "${api_symbols[@]}" | sort) -
))

#将 internal_symbols中的变量名字进一步修改成  --redefine-sym 原函数名=修改后的函数名 的方式,待后面丢给 objcopy 来调用
objcopy_args=($(
    printf "%s\n" "${internal_symbols[@]}" |
    while IFS= read -r sym; do
        new="${prefix_name}${sym}"
        # replace the symbol with a prefix
        echo --redefine-sym "$sym=$new"
        # make the symbol local, future
#        echo -L "$new"
    done
))
#执行更换名字动作
objcopy "${objcopy_args[@]}" $1
done

#$1 是脚本的第一个输入参数,即 libxxx.a

使用方式:

$ redef_func_name.sh libface.a

2.加密内部函数名

如果想隐藏加密内部函数名,可以将上述脚本的改名部分使用下面脚本替代:

bash 复制代码
#在之前的前缀名上,在加入一个前缀, add_() 变成 nicaibudao_nubia_add_(),在经过md5sum处理成"乱码"

more_prefix="nicaibudao_"

md5sum_args=($(
    printf "%s\n" "${private_symbols[@]}" |
    while IFS= read -r sym; do
        new="${more_prefix}$(echo "$sym" | md5sum | cut -d' ' -f1)"
        # replace the symbol with it's md5sum
        echo --redefine-sym "$sym=$new"
        # make the symbol local
        echo -L "$new"
    done
))

objcopy "${md5sum_args[@]}" $1

#$1 是脚本的第一个输入参数,即 libxxx.a

其中api_symbols=( )中要写入本lib的公共的暴露出来的api函数名,否则都不能被调用,而且不知道函数名是什么。

3.小型库重命名

对于小型库,比如nm liblittle.a的信息只有几十或几百行等,可以使用国际友人的如下脚本:
priv.c 源文件中是 内部函数的定义,比如 float add_(float x, float y){ return x+y;}
interface.c 源文件中式 外部api函数定义,比如 float ADD(float a, float b){return add_(a, b);}
如果想隐藏 add_ 并且改名字,而将 ADD暴露出来

bash 复制代码
#!/bin/bash   

namespace="nubia_"

# compile to object files
gcc -c -o priv.o priv.c
gcc -c -o interface.o interface.c

# rename the object file so the names of files are not visible
nofilename="$(echo "nofilenames" | md5sum | cut -d' ' -f1).o"
echo ${nofilename}
ld -relocatable priv.o interface.o -o "$nofilename"

# create the static library
ar rcs static.a "$nofilename"

# list of interface symbols    
public_symbols=( ADD )

# list of private symbols - all symbols except interface symbols
private_symbols=($(
    nm static.a | sed '/^[0-9]\+ T /!d; s///' |
    sort | comm -13 <(printf "%s\n" "${public_symbols[@]}" | sort) -
))

# strip unused symbols, leave only interface symbols
strip_args=($(printf " -K %s " "${public_symbols[@]}"))
strip --strip-unneeded --strip-debug "${strip_args[@]}" static.a

# rename all private symbols with it's md5sum
objcopy_args=($(
    printf "%s\n" "${private_symbols[@]}" |
    while IFS= read -r sym; do
        new="${namespace}$(echo "$sym" | md5sum | cut -d' ' -f1)"
        # replace the symbol with it's md5sum
        echo --redefine-sym "$sym=$new"
        # make the symbol local
        echo -L "$new"
    done
))
objcopy "${objcopy_args[@]}" static.a
相关推荐
赵大仁5 分钟前
在 CentOS 7 上安装 Node.js 20 并升级 GCC、make 和 glibc
linux·运维·服务器·ide·ubuntu·centos·计算机基础
vvw&10 分钟前
Docker Build 命令详解:在 Ubuntu 上构建 Docker 镜像教程
linux·运维·服务器·ubuntu·docker·容器·开源
冷曦_sole36 分钟前
linux-21 目录管理(一)mkdir命令,创建空目录
linux·运维·服务器
最后一个bug38 分钟前
STM32MP1linux根文件系统目录作用
linux·c语言·arm开发·单片机·嵌入式硬件
dessler1 小时前
Docker-Dockerfile讲解(二)
linux·运维·docker
卫生纸不够用1 小时前
子Shell及Shell嵌套模式
linux·bash
world=hello1 小时前
关于科研中使用linux服务器的集锦
linux·服务器
bkspiderx1 小时前
Xshell 和 Xftp 更新提示问题的解决方法及分析
bash·shell·xshell·xftp·破解版本更新
soragui2 小时前
【ChatGPT】OpenAI 如何使用流模式进行回答
linux·运维·游戏