当前位置: 首页 > news >正文

快速读出linux 内核中全局变量

查问题时发现全局变量能读出来会提高效率,于是考虑从怎么读出内核态的全局变量,脚本如下

f = open("/proc/kcore", 'rb')
f.seek(4) # skip magic
assert f.read(1) == b'\x02' # 64 位def read_number(bytes):return int.from_bytes(bytes, 'little', signed=False)elf_header_len = 64
f.seek(elf_header_len - 10)
sec_header_size = read_number(f.read(2))
sec_header_num = read_number(f.read(2))f.seek(elf_header_len + sec_header_size) # ignore note section
sections = []
for i in range(1, sec_header_num):sec_header = f.read(sec_header_size)sections.append({'offset': hex(read_number(sec_header[8:16])),'vaddr': hex(read_number(sec_header[16:24])),'size': hex(read_number(sec_header[32:40])),})print(f"section {i}: " + str(sections[-1]))def addr_to_offset(addr):for sec in sections:vaddr = int(sec['vaddr'], 16)size = int(sec['size'], 16)if addr >= vaddr and addr < vaddr + size:return int(sec['offset'], 16) + (addr - vaddr)raise Exception("ilegel_addr: " + hex(addr))def read_offset_value(offset, type):support_types = ['u8', 'u16', 'u32', 'u64', 's8', 's16', 's32', 's64', 'string','x8','x16','x32','x64']if type not in support_types:raise Exception("type should be in " + str())f.seek(offset)if type == 'string':ret = b''ch = f.read(1)while ch != b'\x00':ret += chch = f.read(1)return retelif type.startswith('s'):return int.from_bytes(f.read(int(type[1:]) // 8), 'little', signed=True)elif type.startswith('u'):return int.from_bytes(f.read(int(type[1:]) // 8), 'little', signed=False)else: # 'x'return hex(int.from_bytes(f.read(int(type[1:]) // 8), 'little', signed=False))def split_to_three_part(path):path = path.strip()prefixes = []suffixes = []prefix_bound = path.find('(')suffix_bound = path.rfind(')')while prefix_bound != -1:prefix = eval(path[:prefix_bound])prefixes.append(prefix)if suffix_bound == -1:raise Exception(f"unmatch backet for {path}")suffix = path[suffix_bound+1:]suffix = eval(suffix) if suffix else 0suffixes.append(suffix)path = path[prefix_bound+1:suffix_bound].strip()prefix_bound = path.find('(')suffix_bound = path.rfind(')')plus_start = path.find('+')if plus_start == -1:plus_start = len(path)minus_start = path.find('-')if minus_start == -1:minus_start = len(path)middle = path[:min(plus_start, minus_start)].strip()middle_part2 = path[len(middle):]middle_part2 = eval(middle_part2) if middle_part2 else 0prefixes.reverse()suffixes.reverse()return prefixes, middle, middle_part2, suffixeswhile True:import sysimport reimport ossys.stdin.flush()msg = input("输入:").strip()try:path, type = msg.split(':')prefixes, middle, middle_part2, suffixes = split_to_three_part(path)if middle.startswith('0x') or re.search(r'[a-z,A-Z]+', middle) is None:start_addr = eval(middle)else: # is variable nameret = os.popen("cat /proc/kallsyms | grep \"" + middle + "\" | awk '{print $1,$3}'").read().strip()if ret == '':raise Exception("no symbol " + middle + " found, please load module first")ret = [i.split(' ') for i in ret.split('\n')]if len(ret) == 1:start_addr = int(ret[0][0], 16)else:find_exact = Falsefor it in ret:if it[1] == start_addr:start_addr = int(it[0], 16)find_exact = Truebreakif not find_exact:print(f"maybe you means:")for it in ret:print(f"  {it[1]}")print(f"find {len(ret)} candidates.")continuestart_offset = addr_to_offset(start_addr + middle_part2)for pre, suf in zip(prefixes, suffixes):start_addr = read_offset_value(start_offset, 'u64')start_offset = pre + addr_to_offset(start_addr) + sufprint(read_offset_value(start_offset, type))except Exception as e:print(e)

输入的格式与 kprobe 的格式类似:+/-偏移(地址)+/-偏移:输出类型
输出类型有:‘u8’, ‘u16’, ‘u32’, ‘u64’, ‘s8’, ‘s16’, ‘s32’, ‘s64’, ‘string’,‘x8’,‘x16’,‘x32’,‘x64’
使用效果如下:
在这里插入图片描述

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • log4j2.xml 使用 application.yml 配置的属性
  • centos部署jar包
  • 【qt】正则表达式来判断是否为邮箱登录
  • 代理模式(大话设计模式)C/C++版本
  • SQL注入安全漏洞与防御策略
  • 展望未来:在【PyCharm】中结合【机器学习】实现高效的图形化处理
  • STM32之六:SysTick系统滴答定时器
  • Jenkins中Node节点与构建任务
  • 32路串口服务器 应用领域
  • ISO 45001:提升职业健康与安全管理水平的关键
  • MAC通过SSH连接VirtualBox中的虚拟机
  • 整洁架构SOLID-里氏替换原则(LSP)
  • MySQL表的内连和外连(重点)
  • Java 如何不用再每次新建线程,直接使用公共线程池
  • IoTDB 集群高效管理:一键启停功能介绍
  • angular2 简述
  • CEF与代理
  • ECMAScript6(0):ES6简明参考手册
  • Flannel解读
  • JavaScript-Array类型
  • Java到底能干嘛?
  • Java精华积累:初学者都应该搞懂的问题
  • laravel5.5 视图共享数据
  • Linux快速配置 VIM 实现语法高亮 补全 缩进等功能
  • MySQL QA
  • react 代码优化(一) ——事件处理
  • spark本地环境的搭建到运行第一个spark程序
  • springMvc学习笔记(2)
  • vue-router的history模式发布配置
  • 大型网站性能监测、分析与优化常见问题QA
  • 汉诺塔算法
  • 浅谈Golang中select的用法
  • 容器服务kubernetes弹性伸缩高级用法
  • 山寨一个 Promise
  • 使用 Xcode 的 Target 区分开发和生产环境
  • 我有几个粽子,和一个故事
  • 验证码识别技术——15分钟带你突破各种复杂不定长验证码
  • 与 ConTeXt MkIV 官方文档的接驳
  • d²y/dx²; 偏导数问题 请问f1 f2是什么意思
  • LevelDB 入门 —— 全面了解 LevelDB 的功能特性
  • 分布式关系型数据库服务 DRDS 支持显示的 Prepare 及逻辑库锁功能等多项能力 ...
  • 交换综合实验一
  • 移动端高清、多屏适配方案
  • ​【C语言】长篇详解,字符系列篇3-----strstr,strtok,strerror字符串函数的使用【图文详解​】
  • ​香农与信息论三大定律
  • ‌JavaScript 数据类型转换
  • #includecmath
  • #Z2294. 打印树的直径
  • #我与Java虚拟机的故事#连载12:一本书带我深入Java领域
  • (读书笔记)Javascript高级程序设计---ECMAScript基础
  • (二十四)Flask之flask-session组件
  • (附源码)计算机毕业设计SSM在线影视购票系统
  • (附源码)流浪动物保护平台的设计与实现 毕业设计 161154
  • (附源码)小程序儿童艺术培训机构教育管理小程序 毕业设计 201740
  • (回溯) LeetCode 131. 分割回文串