xref: /titanic_50/usr/src/lib/libdwarf/common/pro_encode_nm.c (revision f3e7f55e73a39377d55a030f124cc86b3b66a9cc)
1*f3e7f55eSRobert Mustacchi /*
2*f3e7f55eSRobert Mustacchi 
3*f3e7f55eSRobert Mustacchi   Copyright (C) 2000,2004 Silicon Graphics, Inc.  All Rights Reserved.
4*f3e7f55eSRobert Mustacchi 
5*f3e7f55eSRobert Mustacchi   This program is free software; you can redistribute it and/or modify it
6*f3e7f55eSRobert Mustacchi   under the terms of version 2.1 of the GNU Lesser General Public License
7*f3e7f55eSRobert Mustacchi   as published by the Free Software Foundation.
8*f3e7f55eSRobert Mustacchi 
9*f3e7f55eSRobert Mustacchi   This program is distributed in the hope that it would be useful, but
10*f3e7f55eSRobert Mustacchi   WITHOUT ANY WARRANTY; without even the implied warranty of
11*f3e7f55eSRobert Mustacchi   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12*f3e7f55eSRobert Mustacchi 
13*f3e7f55eSRobert Mustacchi   Further, this software is distributed without any warranty that it is
14*f3e7f55eSRobert Mustacchi   free of the rightful claim of any third person regarding infringement
15*f3e7f55eSRobert Mustacchi   or the like.  Any license provided herein, whether implied or
16*f3e7f55eSRobert Mustacchi   otherwise, applies only to this software file.  Patent licenses, if
17*f3e7f55eSRobert Mustacchi   any, provided herein do not apply to combinations of this program with
18*f3e7f55eSRobert Mustacchi   other software, or any other product whatsoever.
19*f3e7f55eSRobert Mustacchi 
20*f3e7f55eSRobert Mustacchi   You should have received a copy of the GNU Lesser General Public
21*f3e7f55eSRobert Mustacchi   License along with this program; if not, write the Free Software
22*f3e7f55eSRobert Mustacchi   Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston MA 02110-1301,
23*f3e7f55eSRobert Mustacchi   USA.
24*f3e7f55eSRobert Mustacchi 
25*f3e7f55eSRobert Mustacchi   Contact information:  Silicon Graphics, Inc., 1500 Crittenden Lane,
26*f3e7f55eSRobert Mustacchi   Mountain View, CA 94043, or:
27*f3e7f55eSRobert Mustacchi 
28*f3e7f55eSRobert Mustacchi   http://www.sgi.com
29*f3e7f55eSRobert Mustacchi 
30*f3e7f55eSRobert Mustacchi   For further information regarding this notice, see:
31*f3e7f55eSRobert Mustacchi 
32*f3e7f55eSRobert Mustacchi   http://oss.sgi.com/projects/GenInfo/NoticeExplan
33*f3e7f55eSRobert Mustacchi 
34*f3e7f55eSRobert Mustacchi */
35*f3e7f55eSRobert Mustacchi 
36*f3e7f55eSRobert Mustacchi 
37*f3e7f55eSRobert Mustacchi 
38*f3e7f55eSRobert Mustacchi #include "config.h"
39*f3e7f55eSRobert Mustacchi #include "libdwarfdefs.h"
40*f3e7f55eSRobert Mustacchi #include <string.h>
41*f3e7f55eSRobert Mustacchi #include "pro_incl.h"
42*f3e7f55eSRobert Mustacchi 
43*f3e7f55eSRobert Mustacchi #define MORE_BYTES      0x80
44*f3e7f55eSRobert Mustacchi #define DATA_MASK       0x7f
45*f3e7f55eSRobert Mustacchi #define DIGIT_WIDTH     7
46*f3e7f55eSRobert Mustacchi #define SIGN_BIT        0x40
47*f3e7f55eSRobert Mustacchi 
48*f3e7f55eSRobert Mustacchi 
49*f3e7f55eSRobert Mustacchi /*-------------------------------------------------------------
50*f3e7f55eSRobert Mustacchi         Encode val as a leb128. This encodes it as an unsigned
51*f3e7f55eSRobert Mustacchi         number.
52*f3e7f55eSRobert Mustacchi ---------------------------------------------------------------*/
53*f3e7f55eSRobert Mustacchi /* return DW_DLV_ERROR or DW_DLV_OK.
54*f3e7f55eSRobert Mustacchi ** space to write leb number is provided by caller, with caller
55*f3e7f55eSRobert Mustacchi ** passing length.
56*f3e7f55eSRobert Mustacchi ** number of bytes used returned thru nbytes arg
57*f3e7f55eSRobert Mustacchi */
58*f3e7f55eSRobert Mustacchi int
_dwarf_pro_encode_leb128_nm(Dwarf_Unsigned val,int * nbytes,char * space,int splen)59*f3e7f55eSRobert Mustacchi _dwarf_pro_encode_leb128_nm(Dwarf_Unsigned val, int *nbytes,
60*f3e7f55eSRobert Mustacchi                             char *space, int splen)
61*f3e7f55eSRobert Mustacchi {
62*f3e7f55eSRobert Mustacchi     char *a;
63*f3e7f55eSRobert Mustacchi     char *end = space + splen;
64*f3e7f55eSRobert Mustacchi 
65*f3e7f55eSRobert Mustacchi     a = space;
66*f3e7f55eSRobert Mustacchi     do {
67*f3e7f55eSRobert Mustacchi         unsigned char uc;
68*f3e7f55eSRobert Mustacchi 
69*f3e7f55eSRobert Mustacchi         if (a >= end) {
70*f3e7f55eSRobert Mustacchi             return DW_DLV_ERROR;
71*f3e7f55eSRobert Mustacchi         }
72*f3e7f55eSRobert Mustacchi         uc = val & DATA_MASK;
73*f3e7f55eSRobert Mustacchi         val >>= DIGIT_WIDTH;
74*f3e7f55eSRobert Mustacchi         if (val != 0) {
75*f3e7f55eSRobert Mustacchi             uc |= MORE_BYTES;
76*f3e7f55eSRobert Mustacchi         }
77*f3e7f55eSRobert Mustacchi         *a = uc;
78*f3e7f55eSRobert Mustacchi         a++;
79*f3e7f55eSRobert Mustacchi     } while (val);
80*f3e7f55eSRobert Mustacchi     *nbytes = a - space;
81*f3e7f55eSRobert Mustacchi     return DW_DLV_OK;
82*f3e7f55eSRobert Mustacchi }
83*f3e7f55eSRobert Mustacchi 
84*f3e7f55eSRobert Mustacchi /* return DW_DLV_ERROR or DW_DLV_OK.
85*f3e7f55eSRobert Mustacchi ** space to write leb number is provided by caller, with caller
86*f3e7f55eSRobert Mustacchi ** passing length.
87*f3e7f55eSRobert Mustacchi ** number of bytes used returned thru nbytes arg
88*f3e7f55eSRobert Mustacchi ** encodes a signed number.
89*f3e7f55eSRobert Mustacchi */
90*f3e7f55eSRobert Mustacchi int
_dwarf_pro_encode_signed_leb128_nm(Dwarf_Signed value,int * nbytes,char * space,int splen)91*f3e7f55eSRobert Mustacchi _dwarf_pro_encode_signed_leb128_nm(Dwarf_Signed value, int *nbytes,
92*f3e7f55eSRobert Mustacchi                                    char *space, int splen)
93*f3e7f55eSRobert Mustacchi {
94*f3e7f55eSRobert Mustacchi     char *str;
95*f3e7f55eSRobert Mustacchi     Dwarf_Signed sign = -(value < 0);
96*f3e7f55eSRobert Mustacchi     int more = 1;
97*f3e7f55eSRobert Mustacchi     char *end = space + splen;
98*f3e7f55eSRobert Mustacchi 
99*f3e7f55eSRobert Mustacchi     str = space;
100*f3e7f55eSRobert Mustacchi 
101*f3e7f55eSRobert Mustacchi     do {
102*f3e7f55eSRobert Mustacchi         unsigned char byte = value & DATA_MASK;
103*f3e7f55eSRobert Mustacchi 
104*f3e7f55eSRobert Mustacchi         value >>= DIGIT_WIDTH;
105*f3e7f55eSRobert Mustacchi 
106*f3e7f55eSRobert Mustacchi         if (str >= end) {
107*f3e7f55eSRobert Mustacchi             return DW_DLV_ERROR;
108*f3e7f55eSRobert Mustacchi         }
109*f3e7f55eSRobert Mustacchi         /*
110*f3e7f55eSRobert Mustacchi          * Remaining chunks would just contain the sign bit, and this chunk
111*f3e7f55eSRobert Mustacchi          * has already captured at least one sign bit.
112*f3e7f55eSRobert Mustacchi          */
113*f3e7f55eSRobert Mustacchi         if (value == sign && ((byte & SIGN_BIT) == (sign & SIGN_BIT))) {
114*f3e7f55eSRobert Mustacchi             more = 0;
115*f3e7f55eSRobert Mustacchi         } else {
116*f3e7f55eSRobert Mustacchi             byte |= MORE_BYTES;
117*f3e7f55eSRobert Mustacchi         }
118*f3e7f55eSRobert Mustacchi         *str = byte;
119*f3e7f55eSRobert Mustacchi         str++;
120*f3e7f55eSRobert Mustacchi     } while (more);
121*f3e7f55eSRobert Mustacchi     *nbytes = str - space;
122*f3e7f55eSRobert Mustacchi     return DW_DLV_OK;
123*f3e7f55eSRobert Mustacchi }
124