Blame src/parsetlv.c

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
}