UDS协议通过0x10(诊断会话控制)服务定义了三级会话体系,包括默认会话(Default Session)、扩展会话(Extended Session)和编程会话(Programming Session)。这套安全体系实现了不同权限级别的分层隔离,确保了诊断和刷写流程的安全性。
![]()
三级安全体系剖析
会话层级
子功能码
“禁区”类比
核心机制
默认会话 0x01
公共区域
ECU上电后的初始状态。这是权限最低的会话,通常只支持信息读取等基础服务。
扩展会话 0x03
工程师工作区
权限更高,可进行参数写入等操作。进入此模式后,通常需要先通过“安全访问”认证。
编程会话 0x02
手术室
最高权限,用于执行固件更新、内存擦写等敏感操作。
注:这些会话与0x27安全访问服务紧密配合,构成了UDS的核心安全防线。会话切换与状态管理
会话切换并非随意进行,它遵循一套严谨的逻辑,并由定时器机制进行管理,防止ECU长时间停留在高权限状态。
标准会话切换路径
一个常见的诊断会话切换路径及报文示例通常如下:
graph TD
Default[默认会话 (Default) 0x10 01]
Default -->|诊断请求| Extended[扩展会话 (Extended) 0x10 03
Request: 10 03
Positive: 50 03]
Extended -->|安全访问| SA{安全访问 (27 Service)}
SA -- 验证通过 --> Extended_Ready[扩展会话(已解锁)]
Extended_Ready -->|编程请求| Programming[编程会话 (Programming) 0x10 02
Request: 10 02
Positive: 50 02]
Default -.->|直接请求失败| x((X))
Programming -.->|不可直接返回| Extended
关键切换逻辑:
路径要求 :必须从 默认会话 切换到 扩展会话 ,再从扩展会话切换到 编程会话 。通常不能从默认会话直接切换到编程会话。
会话保持 :进入扩展或编程会话后,诊断仪需周期性发送 **
TesterPresent服务 (ID:0x3E)**,以维持非默认会话状态,避免因超时而自动退回默认会话。
会话超时是UDS协议中重要的安全机制,其核心时间参数有:
S3 Server time : 非默认会话在无诊断请求时,能保持的最大空闲时间。
P2 Server : ECU 处理单帧诊断请求的典型超时时间。
P2 Server *: 在需要更多处理时间 (如下载/擦除) 时,响应超时时间的最大值。
以下代码示例展示了如何在嵌入式ECU中实现这些核心逻辑(基于常见UDS协议栈实现)。
0x10 服务:会话切换核心处理
当诊断仪发送一个切换会话的请求时(如 10 03),ECU的Dcm_Dcm_ProcessRequest函数会接收到报文,并根据以下逻辑进行处理:
会话超时管理/* 全局变量:用于存储诊断会话状态和应用计时器 */
static Dcm_DiagnoticSessionType Dcm_CurrDiagSession = DCM_DEFAULT_SESSION; // 当前会话状态
static uint16 Dcm_S3ServerTimer = 0; // S3 Server超时计时器,单位ms
static uint8 Dcm_TimerRunning = 0; // 计时器运行标志
/** @brief UDS 0x10 服务的主处理函数
* @param MsgBuf 指向接收到的UDS请求报文的指针
* @param RespBuf 指向响应报文缓冲区的指针(输出参数)
* @return 函数执行状态(通常是E_OK表示成功,E_NOT_OK表示失败)
*/
Std_ReturnType Dcm_DiagnosticSessionControl (uint8* MsgBuf, uint8* RespBuf)
{
uint8 subFunction = MsgBuf[1]; // 获取请求的子功能 (01/02/03)
uint8 retVal = 0; // 返回值
/* 1. 执行通用条件检查 */
retVal = Dcm_PreconditionsCheck(MsgBuf); // 检查是否支持该服务,格式是否正确
if (retVal != NRC_POS_RESP)
{
RespBuf[0] = 0x7F; // 负响应ID
RespBuf[1] = 0x10; // 被拒绝的服务ID
RespBuf[2] = retVal; // 负响应码
return E_OK;
}
/* 2. 根据请求的子功能,执行特定的会话切换逻辑 */
switch (subFunction)
{
case DCM_DEFAULT_SESSION: // 子功能码 0x01 : 切换到默认会话
if (Dcm_CurrDiagSession != DCM_DEFAULT_SESSION)
{
/* 在退出非默认会话前,调用清理函数 */
Dcm_ClearSessionResources(Dcm_CurrDiagSession);
Dcm_CurrDiagSession = DCM_DEFAULT_SESSION;
}
/* 重新初始化默认会话:重置所有诊断相关的临时配置 */
Dcm_ReinitializeDefaultSession();
break;
case DCM_PROGRAMMING_SESSION: // 子功能码 0x02 : 切换到编程会话
/* 检查切换条件1:是否由扩展会话发起 */
if (Dcm_CurrDiagSession != DCM_EXTENDED_DIAGNOSTIC_SESSION)
{
RespBuf[0] = 0x7F;
RespBuf[1] = 0x10;
RespBuf[2] = NRC_7E; // 服务序列中的子功能不支持(当前会话不支持该切换)
return E_OK;
}
/* 检查切换条件2:安全访问是否已解锁(通常需要高安全级别) */
if (Dcm_SecurityLevel != DCM_SEC_LVL_PROGRAMMING) // 假设DCM_SEC_LVL_PROGRAMMING是编程安全级别
{
RespBuf[0] = 0x7F;
RespBuf[1] = 0x10;
RespBuf[2] = NRC_33; // 安全访问未解锁
return E_OK;
}
/* 检查切换条件3:车辆状态是否允许(如车速、电压等) */
if (Dcm_CheckVehicleConditions() != E_OK)
{
RespBuf[0] = 0x7F;
RespBuf[1] = 0x10;
RespBuf[2] = NRC_22; // 条件不满足
return E_OK;
}
/* 执行编程会话的清理和准备工作 */
Dcm_ClearSessionResources(Dcm_CurrDiagSession);
Dcm_CurrDiagSession = DCM_PROGRAMMING_SESSION;
/* 重置并重新配置适用于编程会话的计时参数(如超时间隔) */
Dcm_ConfigureTimingParameters(DCM_PROGRAMMING_SESSION);
break;
case DCM_EXTENDED_DIAGNOSTIC_SESSION: // 子功能码 0x03 : 切换到扩展会话
/* 检查切换条件:是否在默认会话下请求切换 */
if (Dcm_CurrDiagSession != DCM_DEFAULT_SESSION)
{
RespBuf[0] = 0x7F;
RespBuf[1] = 0x10;
RespBuf[2] = NRC_7E;
return E_OK;
}
Dcm_ClearSessionResources(Dcm_CurrDiagSession);
Dcm_CurrDiagSession = DCM_EXTENDED_DIAGNOSTIC_SESSION;
break;
default: // 其他不支持的子功能
RespBuf[0] = 0x7F;
RespBuf[1] = 0x10;
RespBuf[2] = NRC_12; // 子功能不支持
return E_OK;
}
/* 3. 构造0x10服务的肯定响应报文 */
RespBuf[0] = 0x50; // 正响应ID
RespBuf[1] = subFunction; // 回声请求的子功能
RespBuf[2] = (uint8)(Dcm_P2Server_Max >> 8); // 时间参数 P2 Server Max 高位字节
RespBuf[3] = (uint8)(Dcm_P2Server_Max & 0xFF); // 时间参数 P2 Server Max 低位字节
RespBuf[4] = (uint8)(Dcm_P2StarServer_Max >> 8); // 时间参数 P2* Server Max 高位字节
RespBuf[5] = (uint8)(Dcm_P2StarServer_Max & 0xFF);// 时间参数 P2* Server Max 低位字节
RespBuf[6] = 0x01; // 可选参数,通常为0x01
RespBuf[7] = 0xF4; // 可选参数,通常为0xF4
/* 4. 重置S3 Server超时计时器 */
Dcm_S3ServerTimer = Dcm_S3ServerMax; // 重新加载定时器初始值
Dcm_TimerRunning = 1; // 启动计时器return E_OK;
}
ECU的系统Tick中断或后台主循环中,需要持续处理会话的超时逻辑。其核心思想是:当ECU处于某个非默认会话时,会启动一个计时器(S3 Server),并在计时器到期后将会话切回默认会话。
/** @brief 定时器服务函数(通常在1ms或10ms中断/循环中被调用) */
void Dcm_MainFunction_Timer (void)
{
/* 1. 会话超时管理 (S3 Server) */
if (Dcm_TimerRunning == 1) // 如果有计时器正在运行
{
if (Dcm_S3ServerTimer != 0U) // 如果计时器未归零
{
Dcm_S3ServerTimer--; // 计时器递减
}
if (Dcm_S3ServerTimer == 0U) // 如果计时器归零,会话超时
{
Dcm_CurrDiagSession = DCM_DEFAULT_SESSION; // 强制切换回默认会话
Dcm_TimerRunning = 0; // 停止计时器
/* 通知系统:会话已因超时而退出,重置任何与之前会话相关的配置 */
Dcm_NotifySessionTimeout();
}
}/* 2. 其他定时管理,如P2/P2*服务器的超时监控等 */
// ...
}
注意:上述代码为典型设计思路,实际产品级实现通常在AUTOSAR DCM(诊断通信管理)模块的Dcm_DiagnosticSessionControl_SubFnc函数和Dcm_Timer函数中完成。典型安全刷写流程
一个完整的、安全的ECU软件升级流程,会充分体现这套三级安全体系。通常包括以下步骤:
进入扩展会话 :诊断仪发送
10 03,ECU进入扩展会话,解锁部分功能。安全访问认证 :诊断仪通过
27服务(请求种子->发送密钥)通过ECU的高级别身份验证。成功后,ECU解锁编程相关权限。刷写前预检查 :使用
31例程控制服务检查刷写条件(如电压、车速),确保环境安全。进入编程会话 :诊断仪发送
10 02。ECU验证当前在扩展会话且通过安全认证后切换至编程会话,并 停止应用软件运行 ,准备接收新固件。执行固件更新 :通过
34,36,37等服务,将新固件数据下载到ECU的特定内存区域(如Flash)。复位与退出 :更新完成后,发送
11 01硬复位ECU,使其重启。此时ECU会 自动回到默认会话 ,新固件开始运行。
这种分级设计有效确保了即使在安全认证失败的情况下,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.