1. 服务概述
在统一诊断服务(UDS,ISO 14229)中,0x2E 服务(WriteDataByIdentifier)允许外部诊断工具向 ECU(电子控制单元)写入由数据标识符(DID,Data Identifier)指定的单个数据值。典型应用场景包括:
写入车辆 VIN 码(DID 常为
0xF190)设置 ECU 序列号(DID 如
0xF18C)更新软件版本号或标定参数
配置功能选项(如车窗升降比例、灯光延时等)
每个 DID 占 2 字节(16 位),其定义、长度和读写权限由 ECU 制造商在开发阶段确定。部分 DID 允许任意写入,而涉及安全或关键配置的 DID 通常需要先通过0x27 服务(安全访问)解锁权限。
2. 报文格式 2.1 请求报文(诊断工具 → ECU)
字节位置
参数名称
值(例)
0
服务 ID
0x2E
WriteDataByIdentifier
1
DID 高字节
0xF1
数据标识符高位(如 VIN 码标识)
2
DID 低字节
0x90
数据标识符低位
3 … N
数据字节流
0x31 0x32 ...
待写入的原始数据(长度由 DID 定义决定)
示例:向 DID =0xF18C(ECU 序列号)写入 4 字节数据12 34 56 782E F1 8C 12 34 56 78
2.2 肯定响应(ECU → 诊断工具)
字节位置
参数名称
值(例)
0
服务 ID
0x6E
请求服务 ID + 0x40(肯定)
1
DID 高字节
0xF1
原请求中的 DID 高位
2
DID 低字节
0x8C
原请求中的 DID 低位
示例:对上例的肯定响应6E F1 8C
2.3 否定响应(ECU → 诊断工具)
若 ECU 无法执行写入,返回以下格式:
字节位置
参数名称
值(例)
0
否定响应 ID
0x7F
固定值
1
请求服务 ID
0x2E
被拒绝的服务
2
否定响应码
0xXX
NRC(见下文常见代码)
常用否定响应码(NRC):
0x13:报文长度无效(数据长度与 DID 定义不匹配)0x31:请求超出范围(DID 不存在或当前会话不支持)0x33:安全访问被拒绝(未解锁写入权限)0x72:一般编程失败(写入存储时硬件错误)
以下代码模拟了一个简单 ECU,它维护内部 DID 存储表、写入权限映射,并实现了0x2E服务的完整处理流程(含长度校验、权限检查、否定响应生成)。
4. 代码说明与扩展建议#include
#include
#include
#include
// 模拟 ECU 内部存储
class EcuSimulator {
public:
EcuSimulator() {
// 初始化预定义的 DID 及其数据长度、写入权限
// VIN 码 (17 字节,假设初始全为 '?')
dataStore[0xF190] = std::vector(17, '?');
writePerm[0xF190] = true;
// ECU 序列号 (4 字节)
dataStore[0xF18C] = {0x00, 0x00, 0x00, 0x00};
writePerm[0xF18C] = true;
// 只读 DID (2 字节,例如硬件版本)
dataStore[0xF123] = {0x01, 0x02};
writePerm[0xF123] = false;
}
// 处理完整 UDS 请求(假设已剥离掉地址信息等,只保留 UDS 报文)
// 输入:完整 UDS 请求报文(包含服务 ID)
// 输出:响应报文(肯定或否定)
std::vector processRequest(const std::vector& request) {
if (request.empty()) {
return buildNegativeResponse(0x2E, 0x13); // 无效长度
}
uint8_t serviceId = request[0];
if (serviceId == 0x2E) {
// 提取 DID + 数据部分(服务 ID 之后)
std::vector payload(request.begin() + 1, request.end());
return handleWriteDataByIdentifier(payload);
}
// 其他服务忽略,返回不支持
return buildNegativeResponse(serviceId, 0x11);
}
private:
// DID 存储表:DID -> 数据字节数组
std::unordered_map> dataStore;
// 写入权限表:DID -> 是否允许写入
std::unordered_map writePerm;
// 构造否定响应
std::vector buildNegativeResponse(uint8_t reqServiceId, uint8_t nrc) {
return {0x7F, reqServiceId, nrc};
}
// 核心处理函数:处理 WriteDataByIdentifier (0x2E) 服务
// 输入:payload = DID_High + DID_Low + Data
std::vector handleWriteDataByIdentifier(const std::vector& payload) {
// 最小长度:至少包含 DID (2 字节)
if (payload.size() < 2) {
return buildNegativeResponse(0x2E, 0x13); // 报文长度无效
}
// 解析 DID
uint16_t did = (payload[0] << 8) | payload[1];
// 提取待写入的数据(DID 之后的所有字节)
std::vector writeData(payload.begin() + 2, payload.end());
// 1. 检查 DID 是否存在
auto it = dataStore.find(did);
if (it == dataStore.end()) {
return buildNegativeResponse(0x2E, 0x31); // 请求超出范围
}
// 2. 检查写入权限
if (!writePerm[did]) {
return buildNegativeResponse(0x2E, 0x33); // 安全访问被拒绝
}
// 3. 检查写入数据的长度是否与定义长度匹配
if (writeData.size() != it->second.size()) {
return buildNegativeResponse(0x2E, 0x13); // 长度错误
}
// 4. 执行写入操作(这里简单替换,实际可能涉及 EEPROM 写入等)
it->second = writeData;
// 5. 返回肯定响应:0x6E + DID
return {0x6E, payload[0], payload[1]};
}
};
// ---------- 示例使用 ----------
int main() {
EcuSimulator ecu;
// 场景1:向 DID 0xF18C(序列号)写入 4 字节有效数据
std::vector request1 = {0x2E, 0xF1, 0x8C, 0x12, 0x34, 0x56, 0x78};
auto response1 = ecu.processRequest(request1);
std::cout << "Response 1: ";
for (auto b : response1) printf("%02X ", b);
// 预期输出:6E F1 8C
printf("\n");
// 场景2:向只读 DID 0xF123 写入数据(应被拒绝)
std::vector request2 = {0x2E, 0xF1, 0x23, 0xAA, 0xBB};
auto response2 = ecu.processRequest(request2);
std::cout << "Response 2: ";
for (auto b : response2) printf("%02X ", b);
// 预期输出:7F 2E 33
printf("\n");
// 场景3:向存在的 DID 写入错误长度的数据(VIN 需要 17 字节,这里只给 2 字节)
std::vector request3 = {0x2E, 0xF1, 0x90, 0x31, 0x32};
auto response3 = ecu.processRequest(request3);
std::cout << "Response 3: ";
for (auto b : response3) printf("%02X ", b);
// 预期输出:7F 2E 13
printf("\n");return 0;
}
存储模型:实际 ECU 中,DID 数据可能分布在 RAM、EEPROM 或 Flash 中。写入操作可能需要调用底层驱动(如
writeFlash()),本示例简化为主内存副本替换。权限管理:写入权限通常依赖安全访问服务(0x27)。可在处理 0x2E 前检查一个全局安全标志位,若未解锁则返回 NRC
0x33。数据校验:某些 DID 要求数据满足特定范围或格式(如 VIN 需符合 ASCII 规范),可在写入前增加业务校验函数。
多会话支持:UDS 具有不同会话模式(默认会话、扩展会话等)。可增加当前会话状态,限制部分 DID 仅在高权限会话下可写。
事件触发:写入敏感数据后,ECU 可能需要重新初始化相关模块或触发复位。可根据 DID 添加回调机制。
UDS 0x2E 服务为外部工具提供了安全、标准化的 ECU 数据写入能力。开发者需要:
明确定义每个 DID 的数据长度、类型和权限
处理长度不符、DID 不存在、无权限等异常情况并返回规范 NRC
在写入关键数据时配合安全访问机制
上述 C++ 示例展示了服务端处理的核心逻辑,可在此基础上扩展为完整的生产级 UDS 栈模块。
热门跟贴