Fix buffer overflow and fix triple-size UID reported by PN531

A buffer overflow could occur is a triple-size UID card was read with a PN531.
Moreover the way cascade tags were removed was just wrong.

Problem reported by Coverity
CID 1090331 (#1 of 1): Out-of-bounds access (OVERRUN)
10. overrun-buffer-arg: Overrunning buffer pointed to by "&pnti->nai.abtUid[5]" of 10 bytes by passing it to a function which accesses it at byte offset 11 using argument "7UL".

Coverity reported a read out of bounds but actually the real problem if PN531 and triple-size UID will already occur at
    memcpy(pnti->nai.abtUid, pbtRawData, pnti->nai.szUidLen); where abtUid is of size 10 and szUidLen of size 12
This commit is contained in:
Philippe Teuwen 2013-09-19 22:52:50 +02:00
parent 107b4ece8b
commit 3e7dab1e8d

View file

@ -469,6 +469,7 @@ pn53x_decode_target_data(const uint8_t *pbtRawData, size_t szRawData, pn53x_type
nfc_target_info *pnti) nfc_target_info *pnti)
{ {
uint8_t szAttribRes; uint8_t szAttribRes;
const uint8_t *pbtUid;
switch (nmt) { switch (nmt) {
case NMT_ISO14443A: case NMT_ISO14443A:
@ -486,7 +487,7 @@ pn53x_decode_target_data(const uint8_t *pbtRawData, size_t szRawData, pn53x_type
pnti->nai.btSak = *(pbtRawData++); pnti->nai.btSak = *(pbtRawData++);
// Copy the NFCID1 // Copy the NFCID1
pnti->nai.szUidLen = *(pbtRawData++); pnti->nai.szUidLen = *(pbtRawData++);
memcpy(pnti->nai.abtUid, pbtRawData, pnti->nai.szUidLen); pbtUid = pbtRawData;
pbtRawData += pnti->nai.szUidLen; pbtRawData += pnti->nai.szUidLen;
// Did we received an optional ATS (Smardcard ATR) // Did we received an optional ATS (Smardcard ATR)
@ -497,15 +498,19 @@ pn53x_decode_target_data(const uint8_t *pbtRawData, size_t szRawData, pn53x_type
pnti->nai.szAtsLen = 0; pnti->nai.szAtsLen = 0;
} }
// Strip CT (Cascade Tag) to retrieve and store the _real_ UID // For PN531, strip CT (Cascade Tag) to retrieve and store the _real_ UID
// (e.g. 0x8801020304050607 is in fact 0x01020304050607) // (e.g. 0x8801020304050607 is in fact 0x01020304050607)
if ((pnti->nai.szUidLen == 8) && (pnti->nai.abtUid[0] == 0x88)) { if ((pnti->nai.szUidLen == 8) && (pbtUid[0] == 0x88)) {
pnti->nai.szUidLen = 7; pnti->nai.szUidLen = 7;
memmove(pnti->nai.abtUid, pnti->nai.abtUid + 1, 7); memcpy(pnti->nai.abtUid, pbtUid + 1, 7);
} else if ((pnti->nai.szUidLen == 12) && (pnti->nai.abtUid[0] == 0x88) && (pnti->nai.abtUid[4] == 0x88)) { } else if ((pnti->nai.szUidLen == 12) && (pbtUid[0] == 0x88) && (pbtUid[4] == 0x88)) {
pnti->nai.szUidLen = 10; pnti->nai.szUidLen = 10;
memmove(pnti->nai.abtUid, pnti->nai.abtUid + 1, 3); memcpy(pnti->nai.abtUid, pbtUid + 1, 3);
memmove(pnti->nai.abtUid + 3, pnti->nai.abtUid + 5, 7); memcpy(pnti->nai.abtUid + 3, pbtUid + 5, 3);
memcpy(pnti->nai.abtUid + 6, pbtUid + 8, 4);
} else {
// For PN532, PN533
memcpy(pnti->nai.abtUid, pbtUid, pnti->nai.szUidLen);
} }
break; break;