2

我正在使用Microsoft SmartCard API开发 SCM 非接触式 RFID 读卡器。

我所理解的与智能卡交互的通常过程是:
1. 使用 SCardEstablishContext() 建立上下文
2. 使用 SCardListReaders() 列出连接到系统的所有读卡器
3. 选择一个读卡器并等待一张卡插入其中。
4. 一旦检测到卡片,函数 SCardGetStatusChange() 就会返回。
5. 使用 SCardConnect() 通过所选读卡器连接到卡
6. 使用 SCardTransmit() 与卡通话

直到 5 的所有步骤都工作正常,这意味着我得到了正确的卡句柄。可以通过发出 SCardStatus() 函数来进一步验证卡,该函数返回协商的协议和卡的 ATR,这也是正确的。但是当我尝试使用 SCardTransmit() 发送任何 APDU 命令时,它会返回一个错误(错误代码 14,我猜是 SCARD_E_CANT_DISPOSE)。

我不知道出了什么问题或缺少了什么。我尝试了不同的 APDU,例如通过发出 {0xFF、0xCA、0x00、0x00、0x00} 来读取卡 UID 的非常基本的 APDU。我使用 SCM TestResMan 调试工具验证了这个 APDU,它返回了卡的正确 UID。我正在使用 MiFARE 1k 卡和 SCM 非接触式 SCL011 读卡器。

请在下面找到 SCardTransmit() 代码片段(之前的所有函数都返回成功)。

SCARD_IO_REQUEST sioreq;
sioreq.dwProtocol = 2; // SCARD_PROTOCOL_T1
sioreq.cbPciLength = 8;

SCARD_IO_REQUEST rioreq;
rioreq.dwProtocol = 2; // SCARD_PROTOCOL_T1
rioreq.cbPciLength = 8;

byte sendbuffer[256], receivebuffer[256];
ULONG sendbufferLen, receivebufferLen, sizeofUID;

sendbuffer[0] = 0xFF; // CLA - Instruction Class
sendbuffer[1] = 0xCA; // Instruction code
sendbuffer[2] = 0x00; // P1 - 1st parameter
sendbuffer[3] = 0x00; // P2 - 2nd parameter
sendbuffer[4] = 0x00;
sendbufferLen = 0x05;

retval = objScWrapper->SCardTransmit(hCardHandle, &sioreq, sendbuffer,
                                     sendbufferLen, NULL, receivebuffer, 
                                     &receivebufferLen);

如果有人有任何线索或指示,请告诉我...
TIA !

4

2 回答 2

5

您的接收缓冲区大小为 256 字节,但您请求的是 256 字节(Le = 00h),并且接收缓冲区还需要状态字(由两个字节组成,SW1 和 SW2)。

所以我认为你的 14 错误代码是接收缓冲区的通用 OUT_OF_MEMORY 。

您的命令缓冲区也太小,无法容纳 5 字节标题 + 最大 255 字节命令数据。当然,两者肯定都太小而无法处理加长的 ADPU。

Windows 目前对 PCSC 兼容读卡器的智能卡读卡器缓冲区大小非常挑剔。命令缓冲区至少使用 5 + 255 = 260 字节,接收缓冲区至少使用 256 + 2 字节。重复使用缓冲区或安全地处理它们,否则您将发生内存泄漏。

于 2012-03-15T23:10:42.270 回答
0

大多数情况下从这里看起来不错...您可以在问题中回答几个问题,看看我们是否可以到达任何地方...

1) 当您使用 SCardConnect 连接时,您是通过独占、共享还是直接访问进行连接?2)你确定当你连接卡时,卡连接的协议是T1?3) SCARD_IO_REQUEST rioreq 的目的是什么;示例中的行中似乎没有使用此对象?4)我看到你正在通过一个包装器来进行 API 调用。你能分享包装器中的内部 SCardTrasmit 实现吗?

我将根据对您提出的更新问题编辑此答案。

于 2012-03-07T16:51:47.580 回答