1da0c48c4Sopenharmony_ci/* Add signed little endian base 128 integer to a section.
2da0c48c4Sopenharmony_ci   Copyright (C) 2002, 2005 Red Hat, Inc.
3da0c48c4Sopenharmony_ci   This file is part of elfutils.
4da0c48c4Sopenharmony_ci   Written by Ulrich Drepper <drepper@redhat.com>, 2002.
5da0c48c4Sopenharmony_ci
6da0c48c4Sopenharmony_ci   This file is free software; you can redistribute it and/or modify
7da0c48c4Sopenharmony_ci   it under the terms of either
8da0c48c4Sopenharmony_ci
9da0c48c4Sopenharmony_ci     * the GNU Lesser General Public License as published by the Free
10da0c48c4Sopenharmony_ci       Software Foundation; either version 3 of the License, or (at
11da0c48c4Sopenharmony_ci       your option) any later version
12da0c48c4Sopenharmony_ci
13da0c48c4Sopenharmony_ci   or
14da0c48c4Sopenharmony_ci
15da0c48c4Sopenharmony_ci     * the GNU General Public License as published by the Free
16da0c48c4Sopenharmony_ci       Software Foundation; either version 2 of the License, or (at
17da0c48c4Sopenharmony_ci       your option) any later version
18da0c48c4Sopenharmony_ci
19da0c48c4Sopenharmony_ci   or both in parallel, as here.
20da0c48c4Sopenharmony_ci
21da0c48c4Sopenharmony_ci   elfutils is distributed in the hope that it will be useful, but
22da0c48c4Sopenharmony_ci   WITHOUT ANY WARRANTY; without even the implied warranty of
23da0c48c4Sopenharmony_ci   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24da0c48c4Sopenharmony_ci   General Public License for more details.
25da0c48c4Sopenharmony_ci
26da0c48c4Sopenharmony_ci   You should have received copies of the GNU General Public License and
27da0c48c4Sopenharmony_ci   the GNU Lesser General Public License along with this program.  If
28da0c48c4Sopenharmony_ci   not, see <http://www.gnu.org/licenses/>.  */
29da0c48c4Sopenharmony_ci
30da0c48c4Sopenharmony_ci#ifdef HAVE_CONFIG_H
31da0c48c4Sopenharmony_ci# include <config.h>
32da0c48c4Sopenharmony_ci#endif
33da0c48c4Sopenharmony_ci
34da0c48c4Sopenharmony_ci#include <inttypes.h>
35da0c48c4Sopenharmony_ci#include <string.h>
36da0c48c4Sopenharmony_ci
37da0c48c4Sopenharmony_ci#include <libasmP.h>
38da0c48c4Sopenharmony_ci
39da0c48c4Sopenharmony_ci
40da0c48c4Sopenharmony_ciint
41da0c48c4Sopenharmony_ciasm_addsleb128 (AsmScn_t *asmscn, int32_t num)
42da0c48c4Sopenharmony_ci{
43da0c48c4Sopenharmony_ci  if (asmscn == NULL)
44da0c48c4Sopenharmony_ci    return -1;
45da0c48c4Sopenharmony_ci
46da0c48c4Sopenharmony_ci  if (asmscn->type == SHT_NOBITS && unlikely (num != 0))
47da0c48c4Sopenharmony_ci    {
48da0c48c4Sopenharmony_ci      __libasm_seterrno (ASM_E_TYPE);
49da0c48c4Sopenharmony_ci      return -1;
50da0c48c4Sopenharmony_ci    }
51da0c48c4Sopenharmony_ci
52da0c48c4Sopenharmony_ci  if (unlikely (asmscn->ctx->textp))
53da0c48c4Sopenharmony_ci    fprintf (asmscn->ctx->out.file, "\t.sleb128\t%" PRId32 "\n", num);
54da0c48c4Sopenharmony_ci  else
55da0c48c4Sopenharmony_ci    {
56da0c48c4Sopenharmony_ci      char tmpbuf[(sizeof (num) * 8 + 6) / 7];
57da0c48c4Sopenharmony_ci      char *dest = tmpbuf;
58da0c48c4Sopenharmony_ci      uint32_t byte;
59da0c48c4Sopenharmony_ci      int32_t endval = num >> 31;
60da0c48c4Sopenharmony_ci
61da0c48c4Sopenharmony_ci      if (num == 0)
62da0c48c4Sopenharmony_ci	byte = 0;
63da0c48c4Sopenharmony_ci      else
64da0c48c4Sopenharmony_ci	while (1)
65da0c48c4Sopenharmony_ci	  {
66da0c48c4Sopenharmony_ci	    byte = num & 0x7f;
67da0c48c4Sopenharmony_ci
68da0c48c4Sopenharmony_ci	    num >>= 7;
69da0c48c4Sopenharmony_ci	    if (num == endval)
70da0c48c4Sopenharmony_ci	      /* This is the last byte.  */
71da0c48c4Sopenharmony_ci	      break;
72da0c48c4Sopenharmony_ci
73da0c48c4Sopenharmony_ci	    *dest++ = byte | 0x80;
74da0c48c4Sopenharmony_ci	  }
75da0c48c4Sopenharmony_ci
76da0c48c4Sopenharmony_ci      *dest++ = byte;
77da0c48c4Sopenharmony_ci
78da0c48c4Sopenharmony_ci      /* Number of bytes produced.  */
79da0c48c4Sopenharmony_ci      size_t nbytes = dest - tmpbuf;
80da0c48c4Sopenharmony_ci
81da0c48c4Sopenharmony_ci      /* Make sure we have enough room.  */
82da0c48c4Sopenharmony_ci      if (__libasm_ensure_section_space (asmscn, nbytes) != 0)
83da0c48c4Sopenharmony_ci	return -1;
84da0c48c4Sopenharmony_ci
85da0c48c4Sopenharmony_ci      /* Copy the bytes.  */
86da0c48c4Sopenharmony_ci      if (likely (asmscn->type != SHT_NOBITS))
87da0c48c4Sopenharmony_ci	memcpy (&asmscn->content->data[asmscn->content->len], tmpbuf, nbytes);
88da0c48c4Sopenharmony_ci
89da0c48c4Sopenharmony_ci      /* Adjust the pointer in the data buffer.  */
90da0c48c4Sopenharmony_ci      asmscn->content->len += nbytes;
91da0c48c4Sopenharmony_ci
92da0c48c4Sopenharmony_ci      /* Increment the offset in the (sub)section.  */
93da0c48c4Sopenharmony_ci      asmscn->offset += nbytes;
94da0c48c4Sopenharmony_ci    }
95da0c48c4Sopenharmony_ci
96da0c48c4Sopenharmony_ci  return 0;
97da0c48c4Sopenharmony_ci}
98