跳到主要内容

安全单元访问开发指南

简介

电子设备上可能存在一个或多个安全单元(SecureElement,简称SE),比如有eSE(Embedded SE)和SIM卡。安全单元的访问控制,通过GPAC(GlobalPlatform Access Control)规范实现。

场景介绍

应用程序可以通过接口访问安全单元,比如往安全单元里面写入数据,实现在电子设备上模拟一张NFC卡片的目的。该卡片数据可能存储在eSE安全单元,或在SIM卡安全单元上。安全单元上一般会预置有访问控制规则(GPAC规范),应用程序需要具备对应的权限,参考安全单元接口,也就是通过安全单元的访问控制权限校验之后,才能正常访问安全单元。

接口说明

安全单元完整的API说明以及示例代码请参考:安全单元接口

实现安全单元的访问,需要使用到下面的接口。

接口名支持版本功能描述
createService(): Promise<SEService>从API version 12开始支持建立一个可用于连接到系统中所有可用SE的新连接。
getReaders(): Reader[]从API version 10开始支持返回可用SE Reader的数组,包含该设备上支持的所有的安全单元。
openSession(): Session从API version 10开始支持在SE Reader实例上创建连接会话,返回Session实例。
openLogicalChannel(aid: number[]): Promise<Channel>从API version 10开始支持打开逻辑通道,返回逻辑Channel实例对象。
transmit(command: number[]): Promise<number[]>从API version 10开始支持向SE发送APDU数据
close(): void从API version 10开始支持关闭Channel。

主要场景开发步骤

应用程序访问安全单元

  1. import需要的安全单元模块。
  2. 判断设备是否支持安全单元能力。
  3. 访问安全单元,实现数据的读取或写入。
  4. 释放通道资源。

  • 从API version 9之后的应用开发新增支持Stage模型,作为目前主推并长期演进的模型。
  • 由于SE的安全级别较高,必须将构建模式设置为release进行打包,否则应用将无法正常运行。
import { omapi } from '@kit.ConnectivityKit';
import { BusinessError } from '@kit.BasicServicesKit';
import { hilog } from '@kit.PerformanceAnalysisKit';
import { AbilityConstant, UIAbility, Want } from '@kit.AbilityKit';

let seService: omapi.SEService;
let seReaders: omapi.Reader[];
let seSession: omapi.Session;
let seChannel: omapi.Channel;
let testSelectedAid: number[] = [0xA0, 0x00, 0x00, 0x00, 0x03, 0x10, 0x10];
let p2: number = 0x00;

export default class EntryAbility extends UIAbility {
onCreate(want: Want, launchParam: AbilityConstant.LaunchParam) {
hilog.info(0x0000, 'testTag', '%{public}s', 'Ability onCreate');

// 判断设备是否支持安全单元能力
if (!canIUse("SystemCapability.Communication.SecureElement")) {
hilog.error(0x0000, 'testTag', 'secure element unavailable.');
return;
}
hilog.info(0x0000, 'testTag', 'secure element available.');
this.omaTest();
}

private async omaTest() {
// 创建安全单元service,用于访问安全单元
await omapi.createService().then((data) => {
if (data == undefined || !data.isConnected()) {
hilog.error(0x0000, 'testTag', 'secure element service disconnected.');
return;
}
seService = data;
hilog.info(0x0000, 'testTag', 'secure element service connected.');
}).catch((error: BusinessError) => {
hilog.error(0x0000, 'testTag', 'createService error %{public}s', JSON.stringify(error));
return;
});

// 获取设备上所有支持的readers,即所有的安全单元列表
try {
seReaders = seService.getReaders();
} catch (error) {
hilog.error(0x0000, 'testTag', 'getReaders error %{public}s', JSON.stringify(error));
}
if (seReaders == undefined || seReaders.length == 0) {
hilog.error(0x0000, 'testTag', 'no valid reader found.');
seService.shutdown();
return;
}

// 根据业务需求,选择一个安全单元来访问,比如选择eSE或SIM或SIM2,其中SIM2从API version 22开始支持
let reader: (omapi.Reader | undefined);
for (let i = 0; i < seReaders.length; ++i) {
let r = seReaders[i];
// 安全单元的Name来区分,比如是eSE或SIM或SIM2
if (r.getName() === 'SIM') {
reader = r;
break;
}
}
if (reader == undefined) {
hilog.error(0x0000, 'testTag', 'no valid sim reader.');
seService.shutdown();
return;
}
hilog.info(0x0000, 'testTag', 'reader is %{public}s', reader?.getName());

// 在选定的一个安全单元实例上,打开一个会话session
try {
seSession = reader?.openSession() as omapi.Session;
} catch (error) {
hilog.error(0x0000, 'testTag', 'openSession error %{public}s', JSON.stringify(error));
}
if (seSession == undefined) {
hilog.error(0x0000, 'testTag', 'seSession invalid.');
seService.shutdown();
return;
}

// 通过会话session实例,创建逻辑通道或基础通道,一般选择逻辑通道访问,因为基础通道可能是受限的
try {
// testSelectedAid 根据实际业务,修改为打开逻辑通道的应用的aid值
seChannel = await seSession.openLogicalChannel(testSelectedAid, p2);
} catch (exception) {
hilog.error(0x0000, 'testTag', 'openLogicalChannel exception %{public}s', JSON.stringify(exception));
}

if (seChannel == undefined) {
hilog.error(0x0000, 'testTag', 'seChannel invalid.');
seService.shutdown();
return;
}

// 使用通道发送APDU数据到安全单元,testApduData根据实际业务,修改为正确的业务数据值。所填充的APDU数据格式,需要符合APDU规范。
let testApduData = [0x01, 0x02, 0x03, 0x04];
try {
let response: number[] = await seChannel.transmit(testApduData);
hilog.info(0x0000, 'testTag', 'seChannel.transmit() response = %{public}s.', JSON.stringify(response));
} catch (exception) {
hilog.error(0x0000, 'testTag', 'seChannel.transmit() exception = %{public}s.', JSON.stringify(exception));
}

// 通道访问结束后,必须确保通道资源是关闭的
try {
seChannel.close();
} catch (exception) {
hilog.error(0x0000, 'testTag', 'seChannel.close() exception = %{public}s.', JSON.stringify(exception));
}

// 关闭服务资源,关闭应用程序和安全单元服务的绑定关系
seService.shutdown();
}
}