网易首页 > 网易号 > 正文 申请入驻

UDS统一诊断服务详解:核心机制、DoCAN传输与C++代码实现

0
分享至


一、UDS 概述与定位 1.1 什么是UDS?

UDS(Unified Diagnostic Services)是ISO 14229定义的应用层诊断协议,用于汽车电子控制单元(ECU)的故障读取、数据通信、例程控制、软件升级等。

1.2 UDS 与 OBD 的区别

项目

OBD

UDS

目的

排放相关强制诊断

全车ECU增强诊断

范围

有限(发动机、排放)

所有ECU(BCM、GW、BMS等)

扩展性

固定服务

支持自定义服务

网络层

ISO 15765-4

ISO 15765-2(DoCAN)


1.3 DoCAN 协议栈

  • 应用层:UDS(ISO 14229)

  • 网络层:ISO 15765-2(多帧传输)

  • 数据链路层:CAN(ISO 11898)

二、核心通信机制 2.1 请求与响应模型
  • 请求 :SID + SubFunction + 参数

  • 肯定响应 :SID + 0x40 + 参数

  • 否定响应 :0x7F + SID + NRC

示例:
  • 请求读取DID 0xF19022 F1 90

  • 肯定响应: 62 F1 90 01 02 03

  • 否定响应: 7F 22 13 (无效长度)

三、C++ 实现示例(DoCAN + UDS 核心服务) 3.1 数据结构定义

#include  
          
#include
#include
#include

using Byte = uint8_t;
using Bytes = std::vector ;

// UDS 服务 ID(部分)
enum class UDS_SID : Byte {
DIAG_SESSION_CTRL = 0x10,
ECU_RESET = 0x11,
READ_DATA = 0x22,
WRITE_DATA = 0x2E,
SECURITY_ACCESS = 0x27,
TESTER_PRESENT = 0x3E,
ROUTINE_CTRL = 0x31,
READ_DTC = 0x19,
CLEAR_DTC = 0x14
};

// NRC
enum class NRC : Byte {
OK = 0x00,
GENERAL_REJECT = 0x10,
SERVICE_NOT_SUPPORTED = 0x11,
INVALID_FORMAT = 0x13,
CONDITIONS_NOT_CORRECT = 0x22,
SECURITY_ACCESS_DENIED = 0x33,
INVALID_KEY = 0x35
};
3.2 ISO 15765-2 网络层简版封装

// 帧类型
enum class N_PCI_TYPE : Byte {
SINGLE = 0x00,
FIRST = 0x10,
CONSECUTIVE = 0x20,
FLOW_CTRL = 0x30
};

// 单帧/多帧处理(简单示例)
class DoCANTransport {
public:
static Bytes packRequest(const Bytes& udsReq) {
if (udsReq.size() <= 7) {
// 单帧
Byte pci = static_cast (N_PCI_TYPE::SINGLE) | (Byte)udsReq.size();
Bytes frame = {pci};
frame.insert(frame.end(), udsReq.begin(), udsReq.end());
return frame;
} else {
// 简化:实际需拆分为 FF/CF/FC,此处仅示意
std::cout << "[DoCAN] Multi-frame not fully implemented\n";
return {};
}
}

static Bytes unpackResponse(const Bytes& canFrame) {
if (canFrame.empty()) return {};
Byte pci = canFrame[0];
N_PCI_TYPE type = static_cast (pci & 0xF0);
if (type == N_PCI_TYPE::SINGLE) {
int len = pci & 0x0F;
return Bytes(canFrame.begin() + 1, canFrame.begin() + 1 + len);
}
// 实际需重组多帧
return {};
}
};
3.3 UDS 基础处理类

class UDSHandler {
private:
std::map std ::function const Bytes&)>> handlers;
bool securityLocked = true ;

public :
UDSHandler() {
// 注册服务
handlers[UDS_SID::READ_DATA] = [ this ]( const Bytes& req) { return handleReadData(req); };
handlers[UDS_SID::DIAG_SESSION_CTRL] = [ this ]( const Bytes& req) { return handleSessionCtrl(req); };
handlers[UDS_SID::TESTER_PRESENT] = [ this ]( const Bytes& req) { return handleTesterPresent(req); };
}

// 主处理入口
Bytes processRequest(const Bytes& udsReq) {
if (udsReq.empty()) return buildNegativeResponse( 0x00 , NRC::GENERAL_REJECT);
Byte sid = udsReq[ 0 ];
UDS_SID service = static_cast (sid);
if (handlers.find(service) == handlers.end()) {
return buildNegativeResponse(sid, NRC::SERVICE_NOT_SUPPORTED);
}
return handlers[service](udsReq);
}

private :
// 否定响应
Bytes buildNegativeResponse(Byte sid, NRC nrc) {
return { 0x7F , sid, static_cast (nrc)};
}

// 肯定响应
Bytes buildPositiveResponse(Byte sid, const Bytes& data = {}) {
Bytes resp;
resp.push_back(sid + 0x40 );
resp.insert(resp.end(), data.begin(), data.end());
return resp;
}

// 22 读数据
Bytes handleReadData(const Bytes& req) {
if (req.size() < 3 ) return buildNegativeResponse( 0x22 , NRC::INVALID_FORMAT);
Byte didHigh = req[ 1 ];
Byte didLow = req[ 2 ];
uint16_t did = (didHigh << 8 ) | didLow;

// 模拟数据字典
if (did == 0xF190 ) {
return buildPositiveResponse( 0x22 , { 0xF1 , 0x90 , 0x01 , 0x02 , 0x03 });
} else if (did == 0xF187 ) {
return buildPositiveResponse( 0x22 , { 0xF1 , 0x87 , 'V' , '1' , '.' , '0' });
}
return buildNegativeResponse( 0x22 , NRC::GENERAL_REJECT);
}

// 10 会话控制
Bytes handleSessionCtrl(const Bytes& req) {
if (req.size() < 2 ) return buildNegativeResponse( 0x10 , NRC::INVALID_FORMAT);
Byte subfunc = req[ 1 ];
// 抑制肯定响应检查(bit7 = 1 时不发响应)
bool suppressResp = (subfunc & 0x80 ) != 0 ;
Byte session = subfunc & 0x7F ;

std :: cout << "[UDS] Switch to session: " << std ::hex << ( int )session << std :: endl ;
if (!suppressResp) {
return buildPositiveResponse( 0x10 , {subfunc, 0x00 , 0x32 , 0x01 , 0xF4 }); // 仿真实例
}
return {}; // 无响应
}

// 3E 待机握手
Bytes handleTesterPresent(const Bytes& req) {
bool suppress = (req.size() > 1 ) && ((req[ 1 ] & 0x80 ) != 0 );
if (!suppress) {
return buildPositiveResponse( 0x3E , {});
}
return {};
}
};
3.4 主程序示例(诊断请求模拟)

int main() {
UDSHandler uds;
DoCANTransport canLayer;

// 模拟发送:22 F1 90 读取 DID 0xF190
Bytes udsReq = {0x22, 0xF1, 0x90};
Bytes canFrame = DoCANTransport::packRequest(udsReq);

std::cout << "Send CAN: ";
for (auto b : canFrame) printf("%02X ", b);
std::cout << std::endl;

// 模拟 ECU 收到 CAN 帧,解包得到 UDS 请求
Bytes receivedUdsReq = DoCANTransport::unpackResponse(canFrame);
Bytes udsResp = uds.processRequest(receivedUdsReq);

std::cout << "UDS Response: ";
for (auto b : udsResp) printf("%02X ", b);
std::cout << std::endl;

// 测试会话切换(抑制响应)
Bytes sessionReq = {0x10, 0x83}; // subfunc 0x83 = 0x03 + 抑制bit
canFrame = DoCANTransport::packRequest(sessionReq);
receivedUdsReq = DoCANTransport::unpackResponse(canFrame);
udsResp = uds.processRequest(receivedUdsReq);
if (udsResp.empty()) {
std::cout << "[TesterPresent] No response (suppressed)" << std::endl;
}

return 0;
}
输出示例:

Send CAN: 03 22 F1 90 
UDS Response: 62 F1 90 01 02 03
[UDS] Switch to session: 3
[TesterPresent] No response (suppressed)
四、关键服务实现要点 4.1 $27 安全访问(种子密钥简化)

class SecureUDSHandler : public UDSHandler {
int seed = 0x1234;
int expectedKey = 0x5678; // 实际应为算法:key = seed ^ 0x5555


Bytes handleSecurityAccess(const Bytes& req) {
Byte subfunc = req[1];
if (subfunc == 0x05) { // 请求种子
return buildPositiveResponse(0x27, {0x05, (Byte)(seed >> 8), (Byte)seed});
} else if (subfunc == 0x06) { // 发送密钥
int key = (req[2] << 8) | req[3];
if (key == expectedKey) {
securityLocked = false;
return buildPositiveResponse(0x27, {0x06});
}
return buildNegativeResponse(0x27, NRC::INVALID_KEY);
}
return buildNegativeResponse(0x27, NRC::INVALID_FORMAT);
}
};
4.2 14 DTC 处理示例

// 简化 DTC 读取(按状态掩码 0xFF)
Bytes handleReadDTC(const Bytes& req) {
if (req.size() >= 2 && req[1] == 0x02) {
// 模拟返回 2 个 DTC
return {0x59, 0x02,
0x01, 0x23, 0x45, 0x80, // DTC 1 + status
0x01, 0x67, 0x89, 0x20}; // DTC 2
}
return buildNegativeResponse(0x19, NRC::INVALID_FORMAT);
}
五、总结

特性

描述

标准化

UDS 统一了诊断服务格式,减少重复开发

灵活性

支持自定义 DID、Routine、安全算法

可扩展

基于 ISO 15765-2 可承载任意长度诊断数据

工程落地

现代汽车诊断工具(CANoe、PCAN、ZLG)均支持 UDS + DoCAN

本 C++ 示例展示了:

  • UDS 请求/响应处理框架

  • DoCAN 单帧打包/解包

  • 22/$3E 服务的简化实现

  • 否定响应与抑制响应机制

实际 ECU 开发中需完善多帧传输、定时管理、会话状态机和安全访问算法。

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
炸了!美伊战火重开,美军航母突然遭袭!

炸了!美伊战火重开,美军航母突然遭袭!

大嘴说天下
2026-04-30 19:07:44
红场阅兵凉了:去年27国捧,今年只剩4家“自己人”

红场阅兵凉了:去年27国捧,今年只剩4家“自己人”

斯德哥尔摩的帕金森
2026-05-01 12:29:58
爷爷最后一面不见,姥姥录音骂卖国!张本智和,你赢了世界输了家

爷爷最后一面不见,姥姥录音骂卖国!张本智和,你赢了世界输了家

曹老师评球
2026-05-01 13:49:39
悲催!舅妈在舅舅离世次年改嫁,拉黑所有亲属,临终前求合葬被拒

悲催!舅妈在舅舅离世次年改嫁,拉黑所有亲属,临终前求合葬被拒

火山詩话
2026-05-01 09:41:42
湖南男子买烟炫富,致使一家三口被灭,2岁儿子在遗体边躺了两天

湖南男子买烟炫富,致使一家三口被灭,2岁儿子在遗体边躺了两天

莫地方
2026-04-24 00:10:03
“郴州1岁男童被人入室抢走”案:和孩子朝夕相处的奶奶,一度被家人冤枉是拐走孩子的共犯

“郴州1岁男童被人入室抢走”案:和孩子朝夕相处的奶奶,一度被家人冤枉是拐走孩子的共犯

极目新闻
2026-05-01 10:47:56
章若楠空杯到底有多美?网友说:这颜值谁顶得住,难怪都想娶!

章若楠空杯到底有多美?网友说:这颜值谁顶得住,难怪都想娶!

老吴教育课堂
2026-04-30 12:14:14
惨无人道!以军用军犬强奸巴勒斯坦囚犯,全程录像,受害者:想死

惨无人道!以军用军犬强奸巴勒斯坦囚犯,全程录像,受害者:想死

史行途
2026-05-01 12:29:39
143页!人大知名教授被举报:名人之后、9个爱人,聊天内容流出

143页!人大知名教授被举报:名人之后、9个爱人,聊天内容流出

温柔看世界
2026-04-30 11:32:33
李湘在长沙小区被路人偶遇,整个人瘦到像换了个人,忒美了

李湘在长沙小区被路人偶遇,整个人瘦到像换了个人,忒美了

动物奇奇怪怪
2026-04-30 17:30:48
两性关系:55-65岁这十年,惜命最好的方式,不是锻炼,而是这6点

两性关系:55-65岁这十年,惜命最好的方式,不是锻炼,而是这6点

周哥一影视
2026-04-17 06:45:59
海航空姐卖货,真的是太拼了

海航空姐卖货,真的是太拼了

微微热评
2026-05-01 15:58:13
他是水货探花?季后赛得分全队第1,篮板第1,比锡安莫兰特强多了

他是水货探花?季后赛得分全队第1,篮板第1,比锡安莫兰特强多了

球毛鬼胎
2026-05-01 21:10:06
黄奇帆再预言未来房地产,今年已基本应验,明年或大概率又是对的

黄奇帆再预言未来房地产,今年已基本应验,明年或大概率又是对的

巢客HOME
2026-05-01 05:25:03
一动不动!五一最堵6大景点第1名堵到怀疑人生,第5名直接劝退

一动不动!五一最堵6大景点第1名堵到怀疑人生,第5名直接劝退

以茶带书
2026-05-01 18:02:43
我国急需改名的5座城市,起名太随意,连本地人也吐槽名字有点土

我国急需改名的5座城市,起名太随意,连本地人也吐槽名字有点土

长风文史
2026-04-30 22:08:50
曝阿尔瓦雷斯决意离队,或配合巴萨“逼宫”,马竞要价1.5亿欧元

曝阿尔瓦雷斯决意离队,或配合巴萨“逼宫”,马竞要价1.5亿欧元

夏侯看英超
2026-05-01 22:57:00
弘一法师:当你开始冷漠、独来独往、杀伐果断——恭喜,你重生了

弘一法师:当你开始冷漠、独来独往、杀伐果断——恭喜,你重生了

杏花烟雨江南的碧园
2026-04-30 13:15:03
72年周总理正批阅文件,听到消息后厉声问:他死了,为什么瞒着我

72年周总理正批阅文件,听到消息后厉声问:他死了,为什么瞒着我

兴趣知识
2026-05-01 19:01:48
网红白冰成名史:曾是发廊小哥,一条广告66万,奢侈到无法想象

网红白冰成名史:曾是发廊小哥,一条广告66万,奢侈到无法想象

叶公子
2026-04-29 14:19:53
2026-05-01 23:51:00
新能源自动驾驶 incentive-icons
新能源自动驾驶
专注于半导体行业资讯
965文章数 347关注度
往期回顾 全部

科技要闻

DeepSeek发布多模态论文又连夜删除

头条要闻

美国也搞起"人肉代购" "去墨西哥买中国车"教程疯传

头条要闻

美国也搞起"人肉代购" "去墨西哥买中国车"教程疯传

体育要闻

无奈!约基奇:这要在塞尔维亚 全队早被炒了

娱乐要闻

马筱梅产后身材恢复超好 现身户外直播

财经要闻

GPU神话松动,AI真正的战场变了

汽车要闻

限时9.67万起 吉利星越L/星瑞i-HEV智擎混动上市

态度原创

健康
家居
数码
手机
公开课

干细胞治烧烫伤面临这些“瓶颈”

家居要闻

灵动实用 生活艺术场

数码要闻

华硕官宣洛天依“出席”天选2026新品发布会,将有联名新品

手机要闻

华为Pura 90系列、Pura X Max手机备件价格公布

公开课

李玫瑾:为什么性格比能力更重要?

无障碍浏览 进入关怀版