Skip to content
体验新版
项目
组织
正在加载...
登录
切换导航
打开侧边栏
openeuler
openEuler-Advisor
提交
a0ce03b8
O
openEuler-Advisor
项目概览
openeuler
/
openEuler-Advisor
通知
35
Star
4
Fork
4
代码
文件
提交
分支
Tags
贡献者
分支图
Diff
Issue
0
列表
看板
标记
里程碑
合并请求
0
Wiki
0
Wiki
分析
仓库
DevOps
项目成员
Pages
O
openEuler-Advisor
项目概览
项目概览
详情
发布
仓库
仓库
文件
提交
分支
标签
贡献者
分支图
比较
Issue
0
Issue
0
列表
看板
标记
里程碑
合并请求
0
合并请求
0
Pages
分析
分析
仓库分析
DevOps
Wiki
0
Wiki
成员
成员
收起侧边栏
关闭侧边栏
动态
分支图
创建新Issue
提交
Issue看板
体验新版 GitCode,发现更多精彩内容 >>
提交
a0ce03b8
编写于
7月 29, 2020
作者:
S
Shinwell Hu
浏览文件
操作
浏览文件
下载
电子邮件补丁
差异文件
import and improve check_abi
上级
e5d57043
变更
1
显示空白变更内容
内联
并排
Showing
1 changed file
with
330 addition
and
0 deletion
+330
-0
advisors/check_abi.py
advisors/check_abi.py
+330
-0
未找到文件。
advisors/check_abi.py
0 → 100755
浏览文件 @
a0ce03b8
#!/usr/bin/python3
#******************************************************************************
# Copyright (c) Huawei Technologies Co., Ltd. 2020-2020. All rights reserved.
# licensed under the Mulan PSL v2.
# You can use this software according to the terms and conditions of the Mulan PSL v2.
# You may obtain a copy of Mulan PSL v2 at:
# http://license.coscl.org.cn/MulanPSL2
# THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
# PURPOSE.
# See the Mulan PSL v2 for more details.
# Author: wangchuangGG
# Create: 2020-07-20
# ******************************************************************************/
"""
(1) This script is used to check the ABI changes between the old
and new versions of dynamic libraries.
The merged result on difference is saved in the xxx_all_abidiff.out file in the working directory
default path: /var/tmp/xxx_all_abidiff.out
(2) This script depends on abidiff from libabigail package.
(3) Command parameters
This script accept two kind of command: compare_rpm or compare_so
Run it without any paramter prints out help message.
"""
import
argparse
import
subprocess
import
sys
import
os
import
logging
import
io
logging
.
basicConfig
(
format
=
'%(message)s'
,
level
=
logging
.
INFO
)
def
parse_command_line
():
"""Parse the command line arguments."""
parser
=
argparse
.
ArgumentParser
(
prog
=
"check_abi"
)
parser
.
add_argument
(
"-d"
,
"--work_path"
,
default
=
"/var/tmp"
,
nargs
=
"?"
,
help
=
"The work path to put rpm2cpio files and results"
" (e.g. /home/tmp_abidiff default: /var/tmp/)"
)
parser
.
add_argument
(
"-a"
,
"--show_all_info"
,
action
=
"store_true"
,
default
=
False
,
help
=
"show all infos includ changes in member name"
)
subparser
=
parser
.
add_subparsers
(
dest
=
'command_name'
,
help
=
"Compare between two RPMs or two .so files"
)
rpm_parser
=
subparser
.
add_parser
(
'compare_rpm'
,
help
=
"Compare between two RPMs"
)
rpm_parser
.
add_argument
(
"-r"
,
"--rpms"
,
required
=
True
,
nargs
=
2
,
metavar
=
(
'old_rpm'
,
'new_rpm'
),
help
=
"Path or URL of both the old and new RPMs"
)
rpm_parser
.
add_argument
(
"-d"
,
"--debuginfo_rpm"
,
nargs
=
2
,
metavar
=
(
'old_debuginfo_rpm'
,
'new_debuginfo_rpm'
),
required
=
False
,
help
=
"Path or URL of both the old and new debuginfo RPMs, corresponding to compared RPMs."
)
rpm_parser
.
set_defaults
(
func
=
process_with_rpm
)
so_parser
=
subparser
.
add_parser
(
'compare_so'
,
help
=
"Compare between two .so files"
)
so_parser
.
add_argument
(
"-s"
,
"--sos"
,
required
=
True
,
nargs
=
2
,
metavar
=
(
'old_so'
,
'new_so'
),
help
=
"Path or URL of both the old and new .so files"
)
so_parser
.
add_argument
(
"-f"
,
"--debuginfo_path"
,
nargs
=
2
,
required
=
False
,
metavar
=
(
'old_debuginfo_path'
,
'new_debuginfo_path'
),
help
=
"Path or URL of both the old and new debuginfo files, corresponding to compared .so files."
)
so_parser
.
set_defaults
(
func
=
process_with_so
)
config
=
parser
.
parse_args
()
if
config
.
command_name
==
None
:
parser
.
print_help
()
sys
.
exit
(
0
)
else
:
return
config
def
list_so_files
(
path
):
"""
Generate a list of all .so files in the directory.
"""
so_files
=
[]
for
dirpath
,
dirnames
,
files
in
os
.
walk
(
path
):
for
filename
in
files
:
fp
=
os
.
path
.
join
(
dirpath
,
filename
)
if
".so"
in
filename
and
not
os
.
path
.
islink
(
fp
):
#fp = os.path.join(dirpath, filename)
so_files
.
append
(
fp
)
return
so_files
def
find_all_so_file
(
path1
,
path2
):
"""
Try to find all .so files.
"""
all_so_pair
=
{}
old_so
=
list_so_files
(
path1
)
new_so
=
list_so_files
(
path2
)
logging
.
debug
(
"old_so:%s
\n
"
,
old_so
)
logging
.
debug
(
"new_so:%s
\n
"
,
new_so
)
if
old_so
and
new_so
:
for
so_file1
in
old_so
:
for
so_file2
in
new_so
:
base_name1
=
(
os
.
path
.
basename
(
so_file1
)).
split
(
'.'
)[
0
]
base_name2
=
(
os
.
path
.
basename
(
so_file2
)).
split
(
'.'
)[
0
]
if
base_name1
==
base_name2
:
all_so_pair
[
so_file1
]
=
so_file2
else
:
logging
.
info
(
"Not found so files"
)
sys
.
exit
(
0
)
logging
.
debug
(
"all so files:%s
\n
"
,
all_so_pair
)
return
all_so_pair
def
get_rpm2cpio_path
(
work_path
,
abipath
):
"""
Get the path to put so file from rpm
return the path.
"""
fp
=
os
.
path
.
join
(
work_path
,
abipath
)
# FIXME
if
os
.
path
.
isdir
(
fp
):
subprocess
.
run
(
"rm -rf {}"
.
format
(
fp
),
shell
=
True
)
subprocess
.
run
(
"mkdir {}"
.
format
(
fp
),
shell
=
True
)
return
fp
def
get_rpm_path
(
rpm_url
,
dest
):
"""Get the path of rpm package"""
if
os
.
path
.
isfile
(
rpm_url
):
abs_rpmpath
=
os
.
path
.
abspath
(
rpm_url
)
logging
.
debug
(
"rpm exists:%s"
,
abs_rpmpath
)
return
abs_rpmpath
else
:
rpm_name
=
os
.
path
.
basename
(
rpm_url
)
rpm_path
=
os
.
path
.
join
(
dest
,
rpm_name
)
logging
.
debug
(
"downloading %s......"
,
rpm_name
)
subprocess
.
call
([
"curl"
,
rpm_url
,
"-L"
,
"--connect-timeout"
,
"10"
,
"--max-time"
,
"600"
,
"-sS"
,
"-o"
,
rpm_path
])
return
rpm_path
def
do_rpm2cpio
(
rpm2cpio_path
,
rpm_file
):
"""
Exec the rpm2cpio at rpm2cpio_path.
"""
os
.
chdir
(
rpm2cpio_path
)
logging
.
debug
(
"
\n
----working in path:%s----"
,
os
.
getcwd
())
logging
.
debug
(
"rpm2cpio %s"
,
rpm_file
)
subprocess
.
run
(
"rpm2cpio {} | cpio -id > /dev/null 2>&1"
.
format
(
rpm_file
),
shell
=
True
)
def
merge_all_abidiff_files
(
all_abidiff_files
,
work_path
,
rpm_base_name
):
"""
Merge the all diff files to merged_file.
return the merged_file.
"""
merged_file
=
os
.
path
.
join
(
work_path
,
"{}_all_abidiff.out"
.
format
(
rpm_base_name
))
if
os
.
path
.
exists
(
merged_file
):
subprocess
.
run
(
"rm -rf {}"
.
format
(
merged_file
),
shell
=
True
)
ofile
=
open
(
merged_file
,
"a+"
)
for
diff_file
in
all_abidiff_files
:
diff_file_name
=
os
.
path
.
basename
(
diff_file
)
ofile
.
write
(
"---------------diffs in {}:----------------
\n
"
.
format
(
diff_file_name
))
for
txt
in
open
(
diff_file
,
"r"
):
ofile
.
write
(
txt
)
ofile
.
close
()
return
merged_file
def
do_abidiff
(
config
,
all_so_pair
,
work_path
,
base_name
,
debuginfo_path
):
"""
Exec the abidiff and write result to files.
return the abidiff returncode.
"""
if
len
(
all_so_pair
)
==
0
:
logging
.
info
(
"There are no .so files to compare"
)
sys
.
exit
(
0
)
if
debuginfo_path
:
logging
.
debug
(
"old_debuginfo_path:%s
\n
new_debuginfo_path:%s"
,
debuginfo_path
[
0
],
debuginfo_path
[
1
])
with_debuginfo
=
True
else
:
with_debuginfo
=
False
return_code
=
0
all_abidiff_files
=
[]
for
old_so_file
in
all_so_pair
:
new_so_file
=
all_so_pair
[
old_so_file
]
logging
.
debug
(
"begin abidiff between %s and %s"
,
old_so_file
,
new_so_file
)
abidiff_file
=
os
.
path
.
join
(
work_path
,
"{}_{}_abidiff.out"
.
format
(
base_name
,
os
.
path
.
basename
(
new_so_file
)))
if
config
.
show_all_info
:
if
with_debuginfo
:
ret
=
subprocess
.
run
(
"abidiff {} {} --d1 {} --d2 {} "
"--harmless > {}"
.
format
(
old_so_file
,
new_so_file
,
debuginfo_path
[
0
],
debuginfo_path
[
1
],
abidiff_file
),
shell
=
True
)
else
:
ret
=
subprocess
.
run
(
"abidiff {} {} --harmless > {}"
.
format
(
old_so_file
,
new_so_file
,
abidiff_file
),
shell
=
True
)
else
:
if
with_debuginfo
:
ret
=
subprocess
.
run
(
"abidiff {} {} --d1 {} --d2 {} > {} "
"--changed-fns --deleted-fns --added-fns"
.
format
(
old_so_file
,
new_so_file
,
debuginfo_path
[
0
],
debuginfo_path
[
1
],
abidiff_file
),
shell
=
True
)
else
:
ret
=
subprocess
.
run
(
"abidiff {} {} > {} --changed-fns"
" --deleted-fns --added-fns"
.
format
(
old_so_file
,
new_so_file
,
abidiff_file
),
shell
=
True
)
all_abidiff_files
.
append
(
abidiff_file
)
logging
.
info
(
"result write in: %s"
,
abidiff_file
)
return_code
|=
ret
.
returncode
merged_file
=
merge_all_abidiff_files
(
all_abidiff_files
,
work_path
,
base_name
)
logging
.
info
(
"all results writed in: %s"
,
merged_file
)
return
return_code
def
validate_sos
(
config
):
"""
Validate the command arguments
"""
for
so
in
config
.
sos
:
if
not
os
.
path
.
isfile
(
so
)
or
".so"
not
in
so
:
logging
.
error
(
f
"
{
so
}
not exists or not a .so file"
)
sys
.
exit
(
0
)
if
config
.
debuginfo_path
:
for
d
in
config
.
debuginfo_path
:
if
not
os
.
path
.
exists
(
d
):
logging
.
error
(
f
"
{
d
}
not exists"
)
sys
.
exit
(
0
)
def
check_result
(
returncode
):
"""
Check the result of abidiff
"""
ABIDIFF_ERROR_BIT
=
1
if
returncode
==
0
:
logging
.
info
(
"No abidiff found"
)
elif
returncode
&
ABIDIFF_ERROR_BIT
:
logging
.
info
(
"An unexpected error happened"
)
else
:
logging
.
info
(
"Found abidiffs"
)
def
process_with_rpm
(
config
):
"""
Process the file with type of rpm.
"""
work_path
=
config
.
work_path
old_rpm2cpio_path
=
get_rpm2cpio_path
(
work_path
,
"old_abi"
)
new_rpm2cpio_path
=
get_rpm2cpio_path
(
work_path
,
"new_abi"
)
logging
.
debug
(
"old_rpm2cpio_path:%s
\n
new_rpm2cpio_path:%s"
,
old_rpm2cpio_path
,
new_rpm2cpio_path
)
old_rpm
=
get_rpm_path
(
config
.
rpms
[
0
],
old_rpm2cpio_path
)
new_rpm
=
get_rpm_path
(
config
.
rpms
[
1
],
new_rpm2cpio_path
)
logging
.
debug
(
"old_rpm:%s
\n
new_rpm:%s
\n
"
,
old_rpm
,
new_rpm
)
do_rpm2cpio
(
old_rpm2cpio_path
,
old_rpm
)
do_rpm2cpio
(
new_rpm2cpio_path
,
new_rpm
)
if
config
.
debuginfo_rpm
:
old_debuginfo_rpm
=
get_rpm_path
(
config
.
debuginfo_rpm
[
0
],
old_rpm2cpio_path
)
new_debuginfo_rpm
=
get_rpm_path
(
config
.
debuginfo_rpm
[
1
],
new_rpm2cpio_path
)
logging
.
debug
(
"old_debuginfo_rpm:%s
\n
"
"new_debuginfo_rpm:%s"
,
old_debuginfo_rpm
,
new_debuginfo_rpm
)
do_rpm2cpio
(
old_rpm2cpio_path
,
old_debuginfo_rpm
)
do_rpm2cpio
(
new_rpm2cpio_path
,
new_debuginfo_rpm
)
os
.
chdir
(
work_path
)
logging
.
debug
(
"
\n
----begin abidiff working in path:%s----"
,
os
.
getcwd
())
old_so_path
=
os
.
path
.
join
(
old_rpm2cpio_path
,
"usr/lib64"
)
new_so_path
=
os
.
path
.
join
(
new_rpm2cpio_path
,
"usr/lib64"
)
all_so_pairs
=
find_all_so_file
(
old_so_path
,
new_so_path
)
old_debuginfo_path
=
os
.
path
.
join
(
old_rpm2cpio_path
,
"usr/lib/debug"
)
new_debuginfo_path
=
os
.
path
.
join
(
new_rpm2cpio_path
,
"usr/lib/debug"
)
debuginfo_path
=
[
old_debuginfo_path
,
new_debuginfo_path
]
rpm_base_name
=
os
.
path
.
basename
(
new_rpm
).
split
(
'.'
)[
0
]
returncode
=
do_abidiff
(
config
,
all_so_pairs
,
work_path
,
rpm_base_name
,
debuginfo_path
)
check_result
(
returncode
)
return
returncode
def
process_with_so
(
config
):
"""
Process the file with type of .so.
"""
validate_sos
(
config
)
work_path
=
config
.
work_path
all_so_pair
=
{}
so_path
=
list
(
map
(
os
.
path
.
abspath
,
config
.
sos
))
all_so_pair
[
so_path
[
0
]]
=
so_path
[
1
]
os
.
chdir
(
work_path
)
logging
.
debug
(
"
\n
----begin abidiff with .so working in path:%s----"
,
os
.
getcwd
())
so_base_name
=
os
.
path
.
basename
(
old_so_path
).
split
(
'.'
)[
0
]
if
config
.
debuginfo_path
:
debuginfo_path
=
list
(
map
(
os
.
path
.
abspath
,
config
.
debuginfo_path
))
else
:
debuginfo_path
=
None
returncode
=
do_abidiff
(
config
,
all_so_pair
,
work_path
,
so_base_name
,
debuginfo_path
)
check_result
(
returncode
)
return
returncode
def
main
():
"""Entry point for check_abi"""
config
=
parse_command_line
()
ret
=
config
.
func
(
config
)
sys
.
exit
(
ret
)
if
__name__
==
"__main__"
:
main
()
编辑
预览
Markdown
is supported
0%
请重试
或
添加新附件
.
添加附件
取消
You are about to add
0
people
to the discussion. Proceed with caution.
先完成此消息的编辑!
取消
想要评论请
注册
或
登录