rts-sim-testing-service/third_party/axle_device/rssp_channel.go

291 lines
8.0 KiB
Go
Raw Normal View History

2023-11-01 17:46:24 +08:00
package axle_device
2023-11-03 13:04:28 +08:00
import (
2023-11-06 18:10:15 +08:00
"fmt"
2023-11-09 13:05:29 +08:00
"joylink.club/bj-rtsts-server/config"
2023-11-03 13:04:28 +08:00
"joylink.club/bj-rtsts-server/third_party/message"
2023-11-06 18:10:15 +08:00
"joylink.club/bj-rtsts-server/third_party/udp"
2023-11-03 13:04:28 +08:00
"log/slog"
)
2023-11-01 17:46:24 +08:00
2023-11-06 18:10:15 +08:00
// 铁路信号安全通信协议实现
// HandleUserData 回调应用层
type HandleUserData func([]byte)
2023-11-02 18:04:37 +08:00
// RsspChannel 实现rssp通信
type RsspChannel struct {
2023-11-06 18:10:15 +08:00
//udp
udpServer udp.UdpServer
//udp
udpClient udp.UdpClient
//回调应用层
handleUserData HandleUserData
2023-11-03 17:12:52 +08:00
//rssp安全通信配置
2023-11-09 13:05:29 +08:00
config *config.RsspConfig
2023-11-03 17:12:52 +08:00
//rssp时钟
rsspTimer *RsspTimer
//批次编号,发送序列号
2023-11-02 18:04:37 +08:00
sn *message.RsspSn
//安全通道1时间戳
ch1Ts *message.RsspLFSR
//安全通道2时间戳
ch2Ts *message.RsspLFSR
//最近一次接收到的报文的安全通道1时间戳
rcvCh1Ts uint32
//最近一次接收到的报文的安全通道2时间戳
rcvCh2Ts uint32
//最近一次接收到的报文的序列号
rcvSn uint32
2023-11-03 17:12:52 +08:00
//时序校验请求发送记录
sendSseRecord *SseFireRecord
2023-11-02 09:24:11 +08:00
}
2023-11-01 17:46:24 +08:00
2023-11-09 13:05:29 +08:00
func (s *RsspChannel) Init(config *config.RsspConfig) *RsspChannel {
2023-11-06 18:10:15 +08:00
s.config = config
2023-11-07 13:22:47 +08:00
s.rsspTimer = &RsspTimer{t: 0}
s.sn = message.NewRsspSn(1)
s.ch1Ts = message.NewRsspLFSR(message.RsspTsC1, 32, s.config.SID1, false)
s.ch2Ts = message.NewRsspLFSR(message.RsspTsC2, 32, s.config.SID2, false)
2023-11-09 13:05:29 +08:00
return s
2023-11-06 18:10:15 +08:00
}
// Start 启动安全通道
func (s *RsspChannel) Start() {
//
s.udpServer = udp.NewServer(fmt.Sprintf(":%d", s.config.LocalUdpPort), s.handleRsspMsg)
s.udpServer.Listen()
//
s.udpClient = udp.NewClient(fmt.Sprintf("%s:%d", s.config.RemoteIp, s.config.RemoteUdpPort))
}
// Stop 关闭安全通道
func (s *RsspChannel) Stop() {
if s.udpServer != nil {
s.udpServer.Close()
}
if s.udpClient != nil {
s.udpClient.Close()
}
}
2023-11-03 17:12:52 +08:00
// RsspTimer rssp时钟每一个tick周期为config.SendingPeriod
type RsspTimer struct {
t uint64
}
func (s *RsspTimer) tick() {
s.t++
}
func (s *RsspTimer) now() uint64 {
return s.t
}
// SseFireRecord 发送时序校验请求的记录
type SseFireRecord struct {
send *message.RsspSse //已经发送的时序校验请求
rsspTime uint64 //发送时序校验请求时的rssp时间
}
func (s *SseFireRecord) record(send *message.RsspSse, rsspTime uint64) {
s.send = send
s.rsspTime = rsspTime
}
func (s *SseFireRecord) clear() {
s.send = nil
s.rsspTime = 0
}
func (s *SseFireRecord) hasRecord() bool {
return s.send != nil
}
// 处理接收到的rssp报文
2023-11-06 18:10:15 +08:00
// 注意本函数由udp socket 协程执行
2023-11-03 17:12:52 +08:00
func (s *RsspChannel) handleRsspMsg(pack []byte) {
2023-11-01 17:46:24 +08:00
//报文头校验
head := &message.RsspHead{}
if !head.Parse(pack) { //解析报文头失败
2023-11-03 13:04:28 +08:00
slog.Debug("丢弃接收的RSSP报文解析报文头失败")
2023-11-01 17:46:24 +08:00
return
}
if !message.RsspHeadMcCheck(head) { //报文类别检测未通过
2023-11-03 13:04:28 +08:00
slog.Debug("丢弃接收的RSSP报文报文类别检测未通过")
2023-11-01 17:46:24 +08:00
return
}
if !message.RsspHeadPicCheck(head) { //协议交互类别检测未通过
2023-11-03 13:04:28 +08:00
slog.Debug("丢弃接收的RSSP报文协议交互类别检测未通过")
2023-11-01 17:46:24 +08:00
return
}
2023-11-02 18:04:37 +08:00
if !s.config.CheckAddress(head.Sa, head.Da) { //校验报文头中源地址和目的地址是否包含在已配置列表中
2023-11-03 13:04:28 +08:00
slog.Debug("丢弃接收的RSSP报文报文头中源地址或目的地址不在在已配置列表中")
2023-11-01 17:46:24 +08:00
return
}
//报文尾校验
if !message.RsspPackCrc16Check(pack) { //整个报文crc16校验未通过
2023-11-03 13:04:28 +08:00
slog.Debug("丢弃接收的RSSP报文报文尾CRC16校验未通过")
2023-11-01 17:46:24 +08:00
return
}
2023-11-02 09:24:11 +08:00
//解析得到RSD、SSE或SRE
rssp := message.ParseRsspPack(head, pack)
if rssp == nil { //解析具体rssp包失败
2023-11-03 13:04:28 +08:00
slog.Debug("丢弃接收的RSSP报文解析具体类别包失败")
2023-11-02 09:24:11 +08:00
return
}
2023-11-03 13:04:28 +08:00
//处理接收到的具体类别RSSP包
2023-11-02 09:24:11 +08:00
switch rssp.Type() {
case message.RSD_A:
fallthrough
case message.RSD_B:
2023-11-03 13:04:28 +08:00
s.handleRsspRsd(rssp.(*message.RsspRsd))
2023-11-02 09:24:11 +08:00
case message.SSE:
2023-11-03 13:04:28 +08:00
s.handleRsspSse(rssp.(*message.RsspSse))
2023-11-02 09:24:11 +08:00
case message.SSR:
2023-11-03 13:04:28 +08:00
s.handleRsspSsr(rssp.(*message.RsspSsr))
2023-11-02 09:24:11 +08:00
}
2023-11-01 17:46:24 +08:00
}
2023-11-02 18:04:37 +08:00
2023-11-03 13:04:28 +08:00
// 处理接收到的实时安全数据
func (s *RsspChannel) handleRsspRsd(rsd *message.RsspRsd) {
2023-11-03 17:12:52 +08:00
if rsd.Pic != message.PIC_MASTER {
slog.Debug("丢弃接收的RSSP-RSD报文来自备系")
return
}
2023-11-03 13:04:28 +08:00
//序列号校验
//接收的序列号小于最近一次有效序列号则触发SSE时序校验
if rsd.Sn < s.rcvSn {
slog.Debug("丢弃接收的RSSP-RSD报文当前接收RSD的序列号小于最近一次接收的RSD的序列号触发SSE")
s.fireSse(rsd)
return
}
2023-11-03 17:12:52 +08:00
dSn := rsd.Sn - s.rcvSn
if dSn > s.config.Mtv {
2023-11-03 13:04:28 +08:00
slog.Debug("丢弃接收的RSSP-RSD报文当前接收RSD的序列号与最近一次接收的RSD的序列号差值过大触发SSE")
s.fireSse(rsd)
return
}
//SVC校验
2023-11-03 17:12:52 +08:00
c1Crc32 := message.RsspC1Crc32(rsd.Sad)
2023-11-06 11:15:13 +08:00
c1SidTs := c1Crc32 ^ rsd.Svc1 ^ message.SCW_C1 //T(n)
2023-11-03 17:12:52 +08:00
//
c2Crc32 := message.RsspC2Crc32(rsd.Sad)
2023-11-06 11:15:13 +08:00
c2SidTs := c2Crc32 ^ rsd.Svc2 ^ message.SCW_C2 //T(n)
2023-11-03 17:12:52 +08:00
//todo ... SVC校验待完善
_ = c1SidTs
_ = c2SidTs
//校验通过
//记录本次接收RSD的序列号和安全校验通道时间戳
s.rcvSn = rsd.Sn
s.rcvCh1Ts = c1SidTs ^ s.config.SID1
s.rcvCh2Ts = c2SidTs ^ s.config.SID2
//通知应用层接收应用数据
2023-11-06 18:10:15 +08:00
s.handleUserData(rsd.Sad)
2023-11-03 13:04:28 +08:00
}
// 触发时序校正请求
func (s *RsspChannel) fireSse(rsd *message.RsspRsd) {
2023-11-03 17:12:52 +08:00
s.sendSse()
2023-11-03 13:04:28 +08:00
}
// 接收到时序校正请求
func (s *RsspChannel) handleRsspSse(rsd *message.RsspSse) {
}
// 接收到时序校正应答
2023-11-03 17:12:52 +08:00
func (s *RsspChannel) handleRsspSsr(ssr *message.RsspSsr) {
//SSR校验
if !s.sendSseRecord.hasRecord() {
slog.Debug("丢弃接收的RSSP-SSR报文未发起过SSE时序校正请求")
return
}
if s.rsspTimer.t-s.sendSseRecord.rsspTime > uint64(s.config.SsrRsspTimeout) {
slog.Debug("丢弃接收的RSSP-SSR报文等待SSE响应超时")
return
}
if ssr.SeSn != s.sendSseRecord.send.Sn {
slog.Debug("丢弃接收的RSSP-SSR报文SSR与SSE不对应")
return
}
//恢复时序
//todo
2023-11-03 13:04:28 +08:00
}
2023-11-02 18:04:37 +08:00
// 将序列号和时间戳更新到下一个值
func (s *RsspChannel) freshSnTs() {
s.sn.GetAndAdd()
s.ch1Ts.GetAndMove()
s.ch2Ts.GetAndMove()
}
// 发送时序校正应答
func (s *RsspChannel) sendSsr(sse *message.RsspSse) {
ssr := &message.RsspSsr{}
//
ssr.Pic = message.PIC_MASTER
ssr.Mc = message.SSR
ssr.Sa = s.config.SrcAddr
ssr.Da = s.config.DstAddr
ssr.SrSn = s.sn.Get() //当前序列号
ssr.SeSn = sse.Sn
ssr.Tic1 = sse.SeqEnq1 ^ s.config.SID1 ^ s.ch1Ts.Get() ^ s.config.DataVer1
ssr.Tic2 = sse.SeqEnq2 ^ s.config.SID2 ^ s.ch2Ts.Get() ^ s.config.DataVer2
ssr.Dvn = 0x01 //预留固定值
//
rsspPack := ssr.Encode()
s.sendPack(rsspPack)
}
// 发送时序校正请求
func (s *RsspChannel) sendSse() {
sse := &message.RsspSse{}
//
sse.Pic = message.PIC_MASTER
sse.Mc = message.SSE
sse.Sa = s.config.SrcAddr
sse.Da = s.config.DstAddr
//时序校正请求,把最近一次接收到的报文中的序列号时间戳发送给发送方
sse.Sn = s.rcvSn
sse.SeqEnq1 = s.createSeqNeq(s.config.SID1, s.rcvCh1Ts)
sse.SeqEnq2 = s.createSeqNeq(s.config.SID2, s.rcvCh2Ts)
//
rsspPack := sse.Encode()
s.sendPack(rsspPack)
}
// 发送用户数据即通过rssp安全通道发送应用层数据通过rssp的RSD报文发送
func (s *RsspChannel) sendUserData(userData []byte) {
rsd := &message.RsspRsd{}
2023-11-06 18:10:15 +08:00
rsd.Pic = s.config.PicType
if s.config.DeviceA {
2023-11-02 18:04:37 +08:00
rsd.Mc = message.RSD_A
} else {
rsd.Mc = message.RSD_B
}
rsd.Sa = s.config.SrcAddr
rsd.Da = s.config.DstAddr
//
rsd.Sn = s.sn.Get()
rsd.Sdl = uint16(len(userData) + 8)
//安全校验通道SVC_1
crc_c1 := message.RsspC1Crc32(userData)
rsd.Svc1 = s.createSvcCode(crc_c1, s.config.SID1, s.ch1Ts.Get(), message.SCW_C1)
//安全校验通道SVC_2
crc_c2 := message.RsspC2Crc32(userData)
rsd.Svc1 = s.createSvcCode(crc_c2, s.config.SID2, s.ch2Ts.Get(), message.SCW_C2)
rsd.Sad = userData
//
rsspPack := rsd.Encode()
s.sendPack(rsspPack)
}
func (s *RsspChannel) createSvcCode(crc32 uint32, sid uint32, ts uint32, scw uint32) uint32 {
return crc32 ^ sid ^ ts ^ scw
}
func (s *RsspChannel) createSeqNeq(sid uint32, ts uint32) uint32 {
return sid ^ ts
}
// 通过网络发送数据
func (s *RsspChannel) sendPack(rsspPack []byte) {
2023-11-06 18:10:15 +08:00
s.udpClient.Send(rsspPack)
2023-11-02 18:04:37 +08:00
}