Blame iconv/iconv.c

Packit 6c4009
/* Convert characters in input buffer using conversion descriptor to
Packit 6c4009
   output buffer.
Packit 6c4009
   Copyright (C) 1997-2018 Free Software Foundation, Inc.
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
   Contributed by Ulrich Drepper <drepper@cygnus.com>, 1997.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <stddef.h> /* for NULL */
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <iconv.h>
Packit 6c4009
Packit 6c4009
#include <gconv_int.h>
Packit 6c4009
Packit 6c4009
#include <assert.h>
Packit 6c4009
Packit 6c4009
Packit 6c4009
size_t
Packit 6c4009
iconv (iconv_t cd, char **inbuf, size_t *inbytesleft, char **outbuf,
Packit 6c4009
       size_t *outbytesleft)
Packit 6c4009
{
Packit 6c4009
  __gconv_t gcd = (__gconv_t) cd;
Packit 6c4009
  char *outstart = outbuf ? *outbuf : NULL;
Packit 6c4009
  size_t irreversible;
Packit 6c4009
  int result;
Packit 6c4009
Packit 6c4009
  if (__glibc_unlikely (inbuf == NULL || *inbuf == NULL))
Packit 6c4009
    {
Packit 6c4009
      if (outbuf == NULL || *outbuf == NULL)
Packit 6c4009
	result = __gconv (gcd, NULL, NULL, NULL, NULL, &irreversible);
Packit 6c4009
      else
Packit 6c4009
	result = __gconv (gcd, NULL, NULL, (unsigned char **) outbuf,
Packit 6c4009
			  (unsigned char *) (outstart + *outbytesleft),
Packit 6c4009
			  &irreversible);
Packit 6c4009
    }
Packit 6c4009
  else
Packit 6c4009
    {
Packit 6c4009
      const char *instart = *inbuf;
Packit 6c4009
Packit 6c4009
      result = __gconv (gcd, (const unsigned char **) inbuf,
Packit 6c4009
			(const unsigned char *)  (*inbuf + *inbytesleft),
Packit 6c4009
			(unsigned char **) outbuf,
Packit 6c4009
			(unsigned char *) (*outbuf + *outbytesleft),
Packit 6c4009
			&irreversible);
Packit 6c4009
Packit 6c4009
      *inbytesleft -= *inbuf - instart;
Packit 6c4009
    }
Packit 6c4009
  if (outstart != NULL)
Packit 6c4009
    *outbytesleft -= *outbuf - outstart;
Packit 6c4009
Packit 6c4009
  switch (__builtin_expect (result, __GCONV_OK))
Packit 6c4009
    {
Packit 6c4009
    case __GCONV_ILLEGAL_DESCRIPTOR:
Packit 6c4009
      __set_errno (EBADF);
Packit 6c4009
      irreversible = (size_t) -1L;
Packit 6c4009
      break;
Packit 6c4009
Packit 6c4009
    case __GCONV_ILLEGAL_INPUT:
Packit 6c4009
      __set_errno (EILSEQ);
Packit 6c4009
      irreversible = (size_t) -1L;
Packit 6c4009
      break;
Packit 6c4009
Packit 6c4009
    case __GCONV_FULL_OUTPUT:
Packit 6c4009
      __set_errno (E2BIG);
Packit 6c4009
      irreversible = (size_t) -1L;
Packit 6c4009
      break;
Packit 6c4009
Packit 6c4009
    case __GCONV_INCOMPLETE_INPUT:
Packit 6c4009
      __set_errno (EINVAL);
Packit 6c4009
      irreversible = (size_t) -1L;
Packit 6c4009
      break;
Packit 6c4009
Packit 6c4009
    case __GCONV_EMPTY_INPUT:
Packit 6c4009
    case __GCONV_OK:
Packit 6c4009
      /* Nothing.  */
Packit 6c4009
      break;
Packit 6c4009
Packit 6c4009
    default:
Packit 6c4009
      assert (!"Nothing like this should happen");
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  return irreversible;
Packit 6c4009
}