|
Packit |
d7e8d0 |
/* parsetlv.c - ASN.1 TLV functions
|
|
Packit |
d7e8d0 |
* Copyright (C) 2005, 2007, 2008, 2012 g10 Code GmbH
|
|
Packit |
d7e8d0 |
*
|
|
Packit |
d7e8d0 |
* This file is free software; you can redistribute it and/or modify
|
|
Packit |
d7e8d0 |
* it under the terms of the GNU Lesser General Public License as
|
|
Packit |
d7e8d0 |
* published by the Free Software Foundation; either version 2.1 of
|
|
Packit |
d7e8d0 |
* the License, or (at your option) any later version.
|
|
Packit |
d7e8d0 |
*
|
|
Packit |
d7e8d0 |
* This file is distributed in the hope that it will be useful,
|
|
Packit |
d7e8d0 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
Packit |
d7e8d0 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
Packit |
d7e8d0 |
* GNU Lesser General Public License for more details.
|
|
Packit |
d7e8d0 |
*
|
|
Packit |
d7e8d0 |
* You should have received a copy of the GNU Lesser General Public License
|
|
Packit |
d7e8d0 |
* along with this program; if not, see <https://www.gnu.org/licenses/>.
|
|
Packit |
d7e8d0 |
*/
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
#ifdef HAVE_CONFIG_H
|
|
Packit |
d7e8d0 |
# include <config.h>
|
|
Packit |
d7e8d0 |
#endif
|
|
Packit |
d7e8d0 |
#include <stdio.h>
|
|
Packit |
d7e8d0 |
#include <stdlib.h>
|
|
Packit |
d7e8d0 |
#include <string.h>
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
#include "parsetlv.h"
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
/* Simple but pretty complete ASN.1 BER parser. Parse the data at the
|
|
Packit |
d7e8d0 |
address of BUFFER with a length given at the address of SIZE. On
|
|
Packit |
d7e8d0 |
success return 0 and update BUFFER and SIZE to point to the value.
|
|
Packit |
d7e8d0 |
Do not update them on error. The information about the object are
|
|
Packit |
d7e8d0 |
stored in the caller allocated TI structure. */
|
|
Packit |
d7e8d0 |
int
|
|
Packit |
d7e8d0 |
_gpgme_parse_tlv (char const **buffer, size_t *size, tlvinfo_t *ti)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
int c;
|
|
Packit |
d7e8d0 |
unsigned long tag;
|
|
Packit |
d7e8d0 |
const unsigned char *buf = (const unsigned char *)(*buffer);
|
|
Packit |
d7e8d0 |
size_t length = *size;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
ti->cls = 0;
|
|
Packit |
d7e8d0 |
ti->tag = 0;
|
|
Packit |
d7e8d0 |
ti->is_cons = 0;
|
|
Packit |
d7e8d0 |
ti->is_ndef = 0;
|
|
Packit |
d7e8d0 |
ti->length = 0;
|
|
Packit |
d7e8d0 |
ti->nhdr = 0;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!length)
|
|
Packit |
d7e8d0 |
return -1;
|
|
Packit |
d7e8d0 |
c = *buf++; length--; ++ti->nhdr;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
ti->cls = (c & 0xc0) >> 6;
|
|
Packit |
d7e8d0 |
ti->is_cons = !!(c & 0x20);
|
|
Packit |
d7e8d0 |
tag = c & 0x1f;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (tag == 0x1f)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
tag = 0;
|
|
Packit |
d7e8d0 |
do
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
tag <<= 7;
|
|
Packit |
d7e8d0 |
if (!length)
|
|
Packit |
d7e8d0 |
return -1;
|
|
Packit |
d7e8d0 |
c = *buf++; length--; ++ti->nhdr;
|
|
Packit |
d7e8d0 |
tag |= c & 0x7f;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
while (c & 0x80);
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
ti->tag = tag;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (!length)
|
|
Packit |
d7e8d0 |
return -1;
|
|
Packit |
d7e8d0 |
c = *buf++; length--; ++ti->nhdr;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if ( !(c & 0x80) )
|
|
Packit |
d7e8d0 |
ti->length = c;
|
|
Packit |
d7e8d0 |
else if (c == 0x80)
|
|
Packit |
d7e8d0 |
ti->is_ndef = 1;
|
|
Packit |
d7e8d0 |
else if (c == 0xff)
|
|
Packit |
d7e8d0 |
return -1;
|
|
Packit |
d7e8d0 |
else
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
unsigned long len = 0;
|
|
Packit |
d7e8d0 |
int count = (c & 0x7f);
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
if (count > sizeof (len) || count > sizeof (size_t))
|
|
Packit |
d7e8d0 |
return -1;
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
for (; count; count--)
|
|
Packit |
d7e8d0 |
{
|
|
Packit |
d7e8d0 |
len <<= 8;
|
|
Packit |
d7e8d0 |
if (!length)
|
|
Packit |
d7e8d0 |
return -1;
|
|
Packit |
d7e8d0 |
c = *buf++; length--; ++ti->nhdr;
|
|
Packit |
d7e8d0 |
len |= c & 0xff;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
ti->length = len;
|
|
Packit |
d7e8d0 |
}
|
|
Packit |
d7e8d0 |
|
|
Packit |
d7e8d0 |
*buffer = (void*)buf;
|
|
Packit |
d7e8d0 |
*size = length;
|
|
Packit |
d7e8d0 |
return 0;
|
|
Packit |
d7e8d0 |
}
|