ISO/IEC 14443a CRC 计算

CDu*_*tko 2 crc rfid nfc psoc

大家好,我正在为我的 NFC 项目完成固件的最后一部分。我正在尝试与使用 NXP MF0ULx1 MIFARE Ultralight EV1 - 非接触式票证 IC 的 AD-740 NFC 标签进行通信。我使用的 NFC 阅读器是 NXP 的 CLRC663。我正在通过 SPI 使用 PSOC5LP 设备控制 NFC 阅读器。

现在你已经掌握了所有的背景信息,我现在要问我的问题。

对于这个特定的NXP 读取方法,我需要将 Cmd 和 Addr 编码为长度为 2 个字节的 CRC。我可以链接到的数据表引用了 ISO/IEC 14443。在谷歌中搜索将我带到维基百科页面,然后显示 PDF 的四个部分。我假设我需要第 4 部分传输协议部分。唯一的问题是此 PDF 被付费墙阻止。这是故意的吗??

在互联网上四处搜寻我发现了一些代码示例,这些示例可能会阻止我购买 170 美元的 PDF 只是为了查看多项式...

代码示例 1 - 我可以在评论中发布源链接。没有代表在主要职位上做这件事。

    // Calculate an ISO 14443a CRC. Code translated from the code in
    // iso14443a_crc().
    func ISO14443aCRC(data []byte) [2]byte {
        crc := uint32(0x6363)
        for _, bt := range data 
        {
            bt ^= uint8(crc & 0xff)
            bt ^= bt << 4
            bt32 := uint32(bt)
            crc = (crc >> 8) ^ (bt32 << 8) ^ (bt32 << 3) ^ (bt32 >> 4)
         }

       return [2]byte{byte(crc & 0xff), byte((crc >> 8) & 0xff)}
    }
Run Code Online (Sandbox Code Playgroud)

代码示例 2 - 我可以在评论中发布源链接。没有代表在主要职位上做这件事。

    void iso14443a_crc(byte_t* pbtData, size_t szLen, byte_t* pbtCrc)
    {
      byte_t bt;
      uint32_t wCrc = 0x6363;

      do {
        bt = *pbtData++;
        bt = (bt^(byte_t)(wCrc & 0x00FF));
        bt = (bt^(bt<<4));
        wCrc = (wCrc >> 8)^((uint32_t)bt << 8)^((uint32_t)bt<<3)^((uint32_t)bt>>4);
      } while (--szLen);

      *pbtCrc++ = (byte_t) (wCrc & 0xFF);
      *pbtCrc = (byte_t) ((wCrc >> 8) & 0xFF);
    }
Run Code Online (Sandbox Code Playgroud)

现在是我关于所有这些信息的最后一个问题......假设:

CRC 多项式为:0x6363

种子值为:0x00FF

视觉表示可以在这里看到

Nip*_*ipo 5

ISO14443A多项式为0x8408,初始值为0x6363。

  • 我可以告诉你阅读 iso14443-3,但这太厚颜无耻了:) (3认同)
  • CRC_A 是根据 ISO/IEC 13239 (HDLC) 定义的,具有自定义初始值。13239本身只定义了CRC16,维基百科称为CRC-16-CCITT。 (2认同)

Pau*_*sma 5

我在计算 ISO/IEC 14443 的 CRC_A 时找到了一些有用的资源:

对于后代,这里是 libnfc 的实现iso14443a_crc

void
iso14443a_crc(uint8_t *pbtData, size_t szLen, uint8_t *pbtCrc)
{
  uint32_t wCrc = 0x6363;

  do {
    uint8_t  bt;
    bt = *pbtData++;
    bt = (bt ^ (uint8_t)(wCrc & 0x00FF));
    bt = (bt ^ (bt << 4));
    wCrc = (wCrc >> 8) ^ ((uint32_t) bt << 8) ^ ((uint32_t) bt << 3) ^ ((uint32_t) bt >> 4);
  } while (--szLen);

  *pbtCrc++ = (uint8_t)(wCrc & 0xFF);
  *pbtCrc = (uint8_t)((wCrc >> 8) & 0xFF);
}
Run Code Online (Sandbox Code Playgroud)